主页 > MySQL | WEB开发 > 如何建立Mysql索引

如何建立Mysql索引

2011 年 02 月 16 日 没有评论

详见:1.http://www.360doc.com/content/10/1113/10/281812_68949035.shtml#

以下是为出现在where子句的字段建一个索引。为方便讲述,我们先建立一个如下的表。
  Code代码如下:
  CREATE TABLE mytable (
  id serial primary key,
  category_id int not null default 0,
  user_id int not null default 0,
  adddate int not null default 0
  );
  很简单吧,不过对于要说明这个问题,已经足够了。如果你在查询时常用类似以下的语句:
  SELECT * FROM mytable WHERE category_id=1;
  最直接的应对之道,是为category_id建立一个简单的索引:
  CREATE INDEX mytable_categoryid
  ON mytable (category_id);
  OK,搞定?先别高兴,如果你有不止一个选择条件呢?例如:
  SELECT * FROM mytable WHERE category_id=1 AND user_id=2;
  你的第一反应可能是,再给user_id建立一个索引。不好,这不是一个最佳的方法。你可以建立多重的索引。
  CREATE INDEX mytable_categoryid_userid ON mytable (category_id,user_id);
  注意到我在命名时的习惯了吗?我使用”表名_字段1名_字段2名”的方式。你很快就会知道我为什么这样做了。
  现在你已经为适当的字段建立了索引,不过,还是有点不放心吧,你可能会问,数据库会真正用到这些索引吗?测试一下就OK,对于大多数的数据库来说,这是很容易的,只要使用EXPLAIN命令:
  EXPLAIN
  SELECT * FROM mytable
  WHERE category_id=1 AND user_id=2;
  This is what Postgres 7.1 returns (exactly as I expected)
  NOTICE: QUERY PLAN:
  Index Scan using mytable_categoryid_userid on
  mytable (cost=0.00..2.02 rows=1 width=16)
  EXPLAIN
  以上是postgres的数据,可以看到该数据库在查询的时候使用了一个索引(一个好开始),而且它使用的是我创建的第二个索引。看到我上面命名的好处了吧,你马上知道它使用适当的索引了。

接着,来个稍微复杂一点的,如果有个ORDER BY字句呢?不管你信不信,大多数的数据库在使用order by的时候,都将会从索引中受益。
  SELECT * FROM mytable
  WHERE category_id=1 AND user_id=2
  ORDER BY adddate DESC;
  有点迷惑了吧?很简单,就象为where字句中的字段建立一个索引一样,也为ORDER BY的字句中的字段建立一个索引:
  CREATE INDEX mytable_categoryid_userid_adddate
  ON mytable (category_id,user_id,adddate);
  注意: “mytable_categoryid_userid_adddate” 将会被截短为
  ”mytable_categoryid_userid_addda”
  CREATE
  EXPLAIN SELECT * FROM mytable
  WHERE category_id=1 AND user_id=2
  ORDER BY adddate DESC;
  NOTICE: QUERY PLAN:
  Sort (cost=2.03..2.03 rows=1 width=16)
  -> Index Scan using mytable_categoryid_userid_addda
  on mytable (cost=0.00..2.02 rows=1 width=16)
  EXPLAIN
  看看EXPLAIN的输出,好象有点恐怖啊,数据库多做了一个我们没有要求的排序,这下知道性能如何受损了吧,看来我们对于数据库的自身运作是有点过于乐观了,那么,给数据库多一点提示吧。
  为了跳过排序这一步,我们并不需要其它另外的索引,只要将查询语句稍微改一下。这里用的是postgres,我们将给该数据库一个额外的提示–在ORDER BY语句中,加入where语句中的字段。这只是一个技术上的处理,并不是必须的,因为实际上在另外两个字段上,并不会有任何的排序操作,不过如果加入,postgres将会知道哪些是它应该做的。
  EXPLAIN SELECT * FROM mytable
  WHERE category_id=1 AND user_id=2
  ORDER BY category_id DESC,user_id DESC,adddate DESC;
  NOTICE: QUERY PLAN:
  Index Scan Backward using
  mytable_categoryid_userid_addda on mytable
  (cost=0.00..2.02 rows=1 width=16)
  EXPLAIN
  现在使用我们料想的索引了,而且它还挺聪明,知道可以从索引后面开始读,从而避免了任何的排序。

发表评论

电子邮件地址不会被公开。 必填项已用*标注


*

您可以使用这些HTML标签和属性: <a href="" title=""> <abbr title=""> <acronym title=""> <b> <blockquote cite=""> <cite> <code> <del datetime=""> <em> <i> <q cite=""> <strike> <strong>