数据库使用3--索引系列

多数数据库,使用 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 命令,来确认索引的使用状况,及时做相应修改。
相关文章
相关标签/搜索