分类: MySQL

MySQL索引整理

MySQL索引分单列索引和组合索引。单列索引,即一个索引只包含单个列,一个表可以有多个单列索引,但这不是组合索引。组合索引,即一个索引包含多个列。
创建索引时,你需要确保该索引是应用在SQL 查询语句的条件(一般作为 WHERE 子句的条件)。
实际上,索引也是一张表,该表保存了主键与索引字段,并指向实体表的记录。

优点:
MySQL索引的建立对于MySQL的高效运行是很重要的,索引可以大大提高MySQL的检索速度。
打个比方,如果合理的设计且使用索引的MySQL是一辆兰博基尼的话,那么没有设计和使用索引的MySQL就是一个人力三轮车。

缺点:
但过多的使用索引将会造成滥用。因此索引也会有它的缺点:虽然索引大大提高了查询速度,同时却会降低更新表的速度,如对表进行INSERT、UPDATE和DELETE。因为更新表时,MySQL不仅要保存数据,还要保存一下索引文件。
建立索引会占用磁盘空间的索引文件。

创建索引
在执行CREATE TABLE语句时可以创建索引,也可以单独用CREATE INDEX或ALTER TABLE来为表增加索引。

1.ALTER TABLE(添加索引)
添加单个索引的语法:

ALTER TABLE `表名` ADD INDEX `索引名` (`字段名` 顺序) ;
ALTER TABLE `UC_USER_PROPERTY_LEVEL` ADD INDEX `IDX_USER_ID` (`USER_ID` ASC) ;

ALTER TABLE用来添加普通索引、UNIQUE索引或PRIMARY KEY索引。
ALTER TABLE table_name ADD INDEX index_name (column_list)
ALTER TABLE table_name ADD UNIQUE (column_list)
ALTER TABLE table_name ADD PRIMARY KEY (column_list)

其中table_name是要增加索引的表名,column_list指出对哪些列进行索引,多列时各列之间用逗号分隔。索引名index_name可选,缺省时,MySQL将根据第一个索引列赋一个名称。另外,ALTER TABLE允许在单个语句中更改多个表,因此可以在同时创建多个索引。

2.CREATE INDEX(创建索引)
创建单个索引的语法:

create index `索引名` on `表名`(`字段名` 顺序)

CREATE INDEX index_name ON table_name (column_list)
CREATE UNIQUE INDEX index_name ON table_name (column_list)
table_name、index_name和column_list具有与ALTER TABLE语句中相同的含义,索引名不可选。另外,不能用CREATE INDEX语句创建PRIMARY KEY索引。
  1. 删除普通索引
    可利用ALTER TABLE或DROP INDEX语句来删除索引。类似于CREATE INDEX语句,DROP INDEX可以在ALTER TABLE内部作为一条语句处理,语法如下。
DROP INDEX index_name ON talbe_name
ALTER TABLE table_name DROP INDEX index_name
ALTER TABLE table_name DROP PRIMARY KEY

其中,前两条语句是等价的,删除掉table_name中的索引index_name。
第3条语句只在删除PRIMARY KEY索引时使用,因为一个表只可能有一个PRIMARY KEY索引,因此不需要指定索引名。如果没有创建PRIMARY KEY索引,但表具有一个或多个UNIQUE索引,则MySQL将删除第一个UNIQUE索引。
如果从表中删除了某列,则索引会受到影响。对于多列组合的索引,如果删除其中的某列,则该列也会从索引中删除。如果删除组成索引的所有列,则整个索引将被删除。

  1. 查看索引
show index from tblname;
show keys from tblname;

Table 表的名称。

Non_unique 如果索引不能包括重复词,则为0。如果可以,则为1。

Key_name 索引的名称。

Seq_in_index 索引中的列序列号,从1开始。

Column_name 列名称。

Collation 列以什么方式存储在索引中。在MySQL中,有值‘A’(升序)或NULL(无分类)。

Cardinality 索引中唯一值的数目的估计值。通过运行ANALYZE TABLE或myisamchk -a可以更新。基数根据被存储为整数的统计数据来计数,所以即使对于小型表,该值也没有必要是精确的。基数越大,当进行联合时,MySQL使用该索引的机会就越大。

Sub_part 如果列只是被部分地编入索引,则为被编入索引的字符的数目。如果整列被编入索引,则为NULL。

Packed 指示关键字如何被压缩。如果没有被压缩,则为NULL。

Null 如果列含有NULL,则含有YES。如果没有,则该列含有NO。

Index_type 用过的索引方法(BTREE, FULLTEXT, HASH, RTREE)。
  1. 唯一索引
    它与前面的普通索引类似,不同的就是:索引列的值必须唯一,但允许有空值。如果是组合索引,则列值的组合必须唯一。它有以下几种创建方式:

创建唯一索引

CREATE UNIQUE INDEX indexName ON mytable(username(length))
修改表结构添加唯一索引
ALTER table mytable ADD UNIQUE [indexName] (username(length))
创建表的时候直接指定
CREATE TABLE mytable(
ID INT NOT NULL,
username VARCHAR(16) NOT NULL,
UNIQUE [indexName] (username(length))
);
  1. 主键索引
    它是一种特殊的唯一索引,不允许有空值。一般是在建表的时候同时创建主键索引:
CREATE TABLE mytable(ID INT NOT NULL, username VARCHAR(16) NOT NULL, PRIMARY KEY(ID));
修改表结构创建主键ALTER TABLE testalter_tbl ADD PRIMARY KEY (id);
ALTER 命令删除主键ALTER TABLE testalter_tbl DROP PRIMARY KEY;
  1. 创建联合索引的语法:create index 索引名 on 表名(字段名1,字段名2)
    给username和password创建联合索引:
create index t1_username_password on t1(username,password)

例:
TB_BIS_POS_DEVICE -> PROJECT_ID MERCHANT_ID
TB_BIS_POS_HEARTBEAT-> DEVICE_ID ENABLED
TB_BIS_POS_ORDER->CREATE_DATE DEVICE_ID
TB_BIS_MERCHANT->SPACE_ID

create index TB_BIS_POS_DEVICE_t1 on TB_BIS_POS_DEVICE (PROJECT_ID,MERCHANT_ID);
create index TB_BIS_POS_HEARTBEAT_t1 on TB_BIS_POS_HEARTBEAT (DEVICE_ID,ENABLED);
create index TB_BIS_POS_ORDER_t1 on TB_BIS_POS_ORDER (CREATE_DATE,DEVICE_ID);
create index TB_BIS_MERCHANT_t1 on TB_BIS_MERCHANT (SPACE_ID);
  1. 使用索引的注意事项
    使用索引时,有以下一些技巧和注意事项:
1.索引不会包含有NULL值的列
只要列中包含有NULL值都将不会被包含在索引中,复合索引中只要有一列含有NULL值,那么这一列对于此复合索引就是无效的。所以我们在数据库设计时不要让字段的默认值为NULL。
2.使用短索引
对串列进行索引,如果可能应该指定一个前缀长度。例如,如果有一个CHAR(255)的列,如果在前10个或20个字符内,多数值是惟一的,那么就不要对整个列进行索引。短索引不仅可以提高查询速度而且可以节省磁盘空间和I/O操作。
3.索引列排序
MySQL查询只使用一个索引,因此如果where子句中已经使用了索引的话,那么order by中的列是不会使用索引的。因此数据库默认排序可以符合要求的情况下不要使用排序操作;尽量不要包含多个列的排序,如果需要最好给这些列创建复合索引。
4.like语句操作
一般情况下不鼓励使用like操作,如果非使用不可,如何使用也是一个问题。like “%aaa%” 不会使用索引而like “aaa%”可以使用索引。
5.不要在列上进行运算
select * from users where YEAR(adddate)<2016;
将在每个行上进行运算,这将导致索引失效而进行全表扫描,因此我们可以改成:
select * from users where adddate<'2016-01-01';
6.不使用NOT IN和<>操作

更多参考:
http://www.runoob.com/mysql/mysql-index.html

相关文章

发表新评论