MYSQL——索引

索引可以大大提高搜索的效率,假如有十万个数据,没有索引直接遍历查询可能需要5s左右,因为mysql需要频繁的进行磁盘读取,但是有索引就可以很快查到。那么索引的原理是什么呢?

索引原理

mysql的是以page为存储单位的,每个page大小16KB,page结构体中有两个page指针,用来寻找上一个和下一个page结构体,还存储了多条记录,记录按照索引值的大小排列。因此,每个page都存储了一定索引值范围的记录。

mysql中会有很多page结构体,即使使用这样的存储方式依然要遍历很多page,效率依然很低。所以mysql使用目录page结构体,这个结构体不存记录,只存储page结构体的索引范围,可以理解为page结构体的索引。所以要搜索记录就先在目录page寻找对应范围的page,再在page中寻找索引对应的记录。另外,page中也会有记录索引值的目录,为了图的简洁就不画出来了。

如果记录很多,也可以创建多层目录page,一般情况下,2层目录page就足够了。其实这样的结构就是B+树,为什么不使用其他数据结构呢?首先线性结构例如链表显然不可用,其次二叉树或者红黑树这样的树在存储庞大数据时是"瘦高"的,树每高一层就意味着多从磁盘中读一次数据,而B+树是一个多叉树,是"矮胖"的,意味着读取磁盘数据的次数更少。最后,hash结构确实可以用来搜索,但是遇到范围搜索就没有B+树效率高了,因为hash一次只能找一个数据,而B+树的page中存储了相邻索引的多个数据。

索引的种类

索引分为聚簇索引和非聚簇索引。

innoDB使用聚簇索引,叶节点的page中存储的是记录;而MyISAM使用非聚簇索引,叶节点的page中存储的是记录的地址。

frm文件存储表结构,区别就在于innoDB将数据和索引都存在ibd文件,而MyISAM是分开存的,将数据放在MYD文件,将索引放在MYI文件。

索引的使用

索引在使用中也分为多种,主键索引,唯一键索引,普通索引和全文索引。

索引创建

创建主键索引
-- 在创建表的时候,直接在字段名后指定 primary key

bash 复制代码
create table user1(id int primary key, name varchar(30));

-- 在创建表的最后,指定某列或某几列为主键索引

bash 复制代码
create table user2(id int, name varchar(30), primary key(id));

-- 创建表以后再添加主键

bash 复制代码
create table user3(id int, name varchar(30));
alter table user3 add primary key(id);

唯一索引的创建
-- 在表定义时,在某列后直接指定 unique 唯一属性。

bash 复制代码
create table user4(id int primary key, name varchar(30) unique);

-- 创建表时,在表的后面指定某列或某几列为 unique

bash 复制代码
create table user5(id int primary key, name varchar(30), unique(name));
create table user6(id int primary key, name varchar(30));
alter table user6 add unique(name);

如果在某一列建立唯一索引,必须保证这列不能有重复数据
如果一个唯一索引上指定 not null ,等价于主键索引
普通索引的创建

--在表的定义最后,指定某列为索引

bash 复制代码
create table user8(id int primary key,
name varchar(20),
email varchar(30),
index(name) 
);

--创建完表以后指定某列为普通索引

bash 复制代码
create table user9(id int primary key, name varchar(20), email
varchar(30));
alter table user9 add index(name); 

-- 创建一个索引名为 idx_name 的索引

bash 复制代码
create table user10(id int primary key, name varchar(20), email
varchar(30));
create index idx_name on user10(name);

全文索引的创建
当对文章字段或有大量文字的字段进行检索时,会使用到全文索引。 MySQL 提供全文索引机制,但是要求表的存储引擎必须是MyISAM ,而且默认的全文索引支持英文,不支持中文。

复合索引

也可以将多列作为索引,查找索引时虽然显示了两个索引,但是这两个索引是同一个,因为索引名字相同:

索引查询

bash 复制代码
show index from 表名;

除了索引时指定索引名,其他普通索引的名字Key_name都是列名。

索引删除

--删除主键索引

bash 复制代码
alter table 表名 drop primary key;

--删除普通索引

bash 复制代码
--方法一
alter table 表名 drom index 索引名;
--方法二
drop index 索引名 on 表名;
相关推荐
冰红茶兑滴水18 分钟前
MySQL 数据库之库操作
数据库·mysql
从未完美过1 小时前
ClickHouse集成Mysql表引擎跨服务器读表说明
服务器·mysql·clickhouse
Fireworkitte1 小时前
MongoDB
数据库·mongodb
来一杯龙舌兰1 小时前
【MongoDB】MongoDB的聚合(Aggregate、Map Reduce)与管道(Pipline) 及索引详解(附详细案例)
数据库·mongodb·mapreduce·索引·aggregate·pipline
爱吃烤鸡翅的酸菜鱼3 小时前
MySQL初学之旅(1)配置与基础操作
java·数据库·mysql·database
三日看尽长安花5 小时前
【分布式数据库】
数据库·分布式
hummhumm6 小时前
Oracle 第22章:数据仓库与OLAP
java·javascript·后端·python·sql·mysql·database
一 乐9 小时前
家常菜点餐|基于java和小程序的家庭大厨家常菜点餐系统设计与实现(源码+数据库+文档)
java·数据库·小程序·家庭点餐小程序·家庭家常菜点餐
小小不董9 小时前
Oracle OCP认证考试考点详解082系列08
linux·运维·服务器·数据库·oracle·dba
zwm_yy10 小时前
ubantu lnmp
运维·mysql·lnmp·ubantu