多数数据库,使用 B 树(Balance Tree)的结构来保存索引。
www.2cto.com
B 树,
最上层节点:根节点
最下层节点:叶子节点
二者之间的节点:中间节点
B 树,显著特征:从根节点,到各个叶子节点的距离都是相等的。如此,检索任何值时,都通过相同数目的节点。
[sql]
CREATE INDEX idx_lname_pinyin ON employee(lname_pinyin);
SHOW INDEX FROM employee\G
DROP INDEX idx_lname_pinyin ON employee;
伴随主键的定义而建立的特别索引,被称为:丛生索引(Clustered Index)
一个表,只有一个丛生索引。
普通的索引,在叶子节点中保存的是指向实际表的指针。
而,丛生索引,在叶子节点中保存的就是实际数据。
丛生索引,
不须要为保存索引,而使用专用的硬盘空间,节约资源
不须要检查索引后,再访问实际的表,提升了效率
建立丛生索引时,须要对表中数据进行排序,所以,在执行数据插入、更新、删除等操做时,比普通索引慢。
复合索引
[sql]
CREATE INDEX idx_pinyin ON employee( lname_pinyin, fname_pinyin);
SHOW INDEX FROM employee\G
惟一性索引
使用 UNIQUE 关键字,来建立不可重复的索引,称为:惟一性索引。
对特定列建立惟一性索引,至关于对该列追加了惟一性制约。
建立惟一性索引的时候,若是对象列中,已经含有重复数据,则:建立失败,报错。
建立成功后,若是,插入重复数据,则:插入失败,报错。
指定多个列,来建立惟一性索引,只要,这些列的组合数据不重复,就能够建立成功。
[sql]
CREATE UNIQUE INDEX idx_fname ON employee(fname);
CREATE UNIQUE INDEX idx_lname ON employee(lname);
CREATE UNIQUE INDEX idx_fullname ON employee(lname, fname);
[sql]
EXPLAIN SELECT * FROM employee WHERE lname_pinyin='wang'\G
[sql]
CREATE INDEX idx_lname_pinyin ON employee(lname_pinyin);
EXPLAIN SELECT * FROM employee WHERE lname_pinyin='wang'\G
EXPLAIN 命令的说明:P100 表格
若是,建立索引后,遍历的次数与建立索引以前,变化不大。
说明,建立索引时,选择的列名不合理,须要选择合适的列重建索引。这是分析索引优劣的方法。
追加了索引后,也不能保证在每次检索时都会使用列索引。
若是, SQL 检索语句编写不当,就没法使用索引。
一、LIKE 运算符,进行模糊检索时,只能在前方一致的检索时,才能使用索引。
如下写法,索引不会被使用。
[sql]
SELECT * FROM employee WHERE lname_pinyin LIKE '%w%';
SELECT * FROM employee WHERE lname_pinyin LIKE '%w';
二、使用 IS NOT NULL 、 <> 的场合,也不会使用索引。以下:
[sql]
SELECT * FROM employee WHERE lname_pinyin IS NOT NULL;
SELECT * FROM employee WHERE lname_pinyin <> 'wang';
三、对列使用了运算或者函数的状况下,不会使用索引,以下:
[sql]
SELECT * FROM employee WHERE YEAR(birth) = '1980'; -- 不使用索引
SELECT * FROM employee WHERE birth >= '1980-01-01' AND birth <= '1980-12-31'; --使用了索引
四、复合索引的第一列,没有包含在 WHERE 条件语句中,以下:
[sql]
CREATE INDEX idx_pinyin ON employee(lname_pinyin, fname_pinyin);
SELECT * FROM employee WHERE lname_pinyin = 'wang' AND fname_pinyin = 'xiao'; --用了索引
SELECT * FROM employee WHERE lname_pinyin = 'wang'; --用了索引
SELECT * FROM employee WHERE fname_pinyin = 'xiao'; -- 没有使用索引
SELECT * FROM employee WHERE lname_pinyin = 'wang' OR fname_pinyin = 'xiao'; -- 没有使用索引
实际开发过程当中,须要积极使用 EXPLAIN 命令,来确认索引的使用状况,及时做相应修改。