Myisam与innodb引擎,索引文件的区别:innodb的次索引指向对主键的引用。myisam的次索引和主索引都指向物理行。myisam一行一行的插入,会产生一行一行的文件,磁盘上有数据文件。tree树的值是磁盘上物理位置的指针。比如加了主键索引,索引排成一棵树的形状。首先根据id=7在主键索引的树上查找,查找到7之后就知道了7所在的物理行,然后就可以找到id=7的那一行数据了。还有一个cat_id索引,根据cat_id=15可以找到数据所在的物理行。所以说myisam的次索引和主索引都指向物理行。
innodb的主键索引,数的每一个叶子下面,直接挂在了每行的数据,id=5的地方挂载的就是id=5的这行数据。数据就在叶子上,不用去磁盘上面查找。如果还有其他索引username,username=li的叶子下面放的是id=7.根据username索引这棵树上找到id=7然后在主键树上找到数据。所以innodb的次索引指向对主键的引用。id的主索引成为聚簇索引,好处是根据主键查非常快,坏处是根据其他索引找的时候要多找一次主键这棵树。username是非聚簇索引。
innodb的主索引文件上 直接存放该行数据,称为聚簇索引,次索引指向对主键的引用。myisam中, 主索引和次索引,都指向物理行(磁盘位置)。注意: innodb来说, 1: 主键索引 既存储索引值,又在叶子中存储行的数据2: 如果没有主键, 则会Unique key做主键 3: 如果没有unique,则系统生成一个内部的rowid做主键.4: 像innodb中,主键的索引结构中,既存储了主键值,又存储了行数据,这种结构称为"聚簇索引"聚簇索引 优势: 根据主键查询条目比较少时,不用回行(数据就在主键节点下)劣势: 如果碰到不规则数据插入时,造成频繁的页分裂.myisam中对于索引文件是要放在内存中缓存起来的。节点会分裂:原来19的节点后来来了15和13,则19的位置换成13,并在下面添加15,19。对于聚簇索引就很严重。对于myisam没什么,对于innodb就很麻烦。
高性能索引策略对于innodb而言,因为节点下有数据文件,因此节点的分裂将会比较慢.对于innodb的主键,尽量用整型,而且是递增的整型.如果是无规律的数据,将会产生的页的分裂,影响速度.
create table A{ id varchar(64) primary key, ver int,}在id和ver上有联合索引10000条数据。为什么select id from A order by id很慢而select id from A order by id,ver很快 如果用的是myisam,那么都用到了索引覆盖,应该是一样都很快,有可能不实用的myisam引擎。myisam无论使用什么索引都是指向物理行的位置。如果是innodb引擎,每个叶子下面直接放的数据,这些数据比较大内存放不下,就放在磁盘上。innodb的主键是聚簇索引。有比较长的列,聚簇索引导致沿id排序时要跨好多块。而且块比较多。所以查找很慢。第二句是联合索引,联合索引没有放数据块(除了主键索引其余索引都指向主键索引,不带数据),而是放的是主键索引的位置指向id的值,不带有数据,文件比较小可以在内存中存放。现在只是取出id不用回行,就是在索引文件中取,而且索引文件比较小就放在内存中,所以很快。第一个语句,也只是在索引文件中查找,发生了索引覆盖,但是这个主键索引文件比较大,而且不一定在内存中,查找主键树的时候来回跳跃就很慢。如果把数据比较大的字段去掉,速度也会提升,因为查找主键索引文件来回跳的时候就不会慢了。
通过下面的规律可以看出-----1: innodb的buffer_page 很强大.2: 聚簇索引的主键值,应尽量是连续增长的值,而不是要是随机值, (不要用随机字符串或UUID)否则会造成大量的页分裂与页移动.
实验: 聚簇索引使用随机值导致页频繁分裂影响速度过程:建立innodb表, 利用php连接mysql,分别规则插入10000条数据,不规则插入10000条数据观察时间的差异,体会聚簇索引,页分裂的影响. create table t5(id int primary key,c1 varchar(500),c2 varchar(500),c3 varchar(500),c4 varchar(500),c5 varchar(500),c6 varchar(500)) engine innodb charset utf8;create table t6(id int primary key,c1 varchar(500),c2 varchar(500),c3 varchar(500),c4 varchar(500),c5 varchar(500),c6 varchar(500)) engine innodb charset utf8;// testinnodb.php$time_start = microtime_float();$str = str_repeat('hello',100);for($i=1;$i<=10000;$i++) { $sql = "insert into t5 values ($i,'$str' , '$str' , '$str' , '$str' , '$str' , '$str')"; //echo $sql; mysql_query($sql , $conn);}$time_end = microtime_float();echo 'seq insert cost' , ($time_end - $time_start) , "seconds\n";function microtime_float(){ list($usec, $sec) = explode(" ", microtime()); return ((float)$usec + (float)$sec);}// rndinnodb.php$base = range(1,10000);shuffle($base);$time_start = microtime_float();$str = str_repeat('hello',100);foreach($base as $i) { $sql = "insert into t6 values ($i,'$str' , '$str' , '$str' , '$str' , '$str' , '$str')"; //echo $sql; mysql_query($sql , $conn);}$time_end = microtime_float();echo 'rand insert cost' , ($time_end - $time_start) , "seconds\n";function microtime_float(){ list($usec, $sec) = explode(" ", microtime()); return ((float)$usec + (float)$sec);}
字段数 | 混乱程度(步长) | 顺序1000条(秒数) | 乱序1000条(秒数) | 顺序写入page页数 | 乱序写入page数 |
1 | 1 | 54.365 | 53.438 | 62 | 91 |
10 | 1 | 53.413 | 62.940 | 235 | 1301 |
10 | 100 |
| 64.18 |
| 1329 |
10 | 1000 |
| 67.512 |
| 1325 |
通过上面的规律可以看出-----1: innodb的buffer_page 很强大.2: 聚簇索引的主键值,应尽量是连续增长的值,而不是要是随机值, (不要用随机字符串或UUID)否则会造成大量的页分裂与页移动.