聚集索引和非聚集索引的区别_主键,聚集索引,不是一个东西

练习一:建表时,可不可以不声明主键?

(1) create table user(

name varchar(10)

)engine=innodb;

(2) insert into user values('shenjian');

(3) insert into user values('shenjian');

画外音:建表时,不声明主键,插入两个相同的元素。

提问,连续执行上面的语句,执行结果是:

A 建表语句(1)报错

B 插入语句(2)报错

C 插入语句(3)报错

D 均不报错

ec1f7122bbf5a8a6c1ba9a2e291df30e.png

实操如上,所以答案是【D 均不报错】

InnoDB必须要有聚集索引,行记录按照聚集索引物理上排序。必须要有聚集索引,并不代表一定要有主键。

(1)如果定义了主键,主键就是聚集索引;

(2)如果没有定义主键,第一个非空(not null)且唯一(unique)列是聚集索引;

(3)如果没有符合条件的列,会自动创建一个隐藏的row-id作为聚集索引;

画外音:本例就属于第三种情况。

练习二:建表时,可不可以不声明主键非空?

(1) create table user(

id int,

name varchar(10),

primary key(id)

)engine=innodb;

(2) insert into user(name) values('shenjian');

(3) insert into user(name) values('shenjian');

画外音:建表时,不声明非空,插入两个相同的元素。

提问,连续执行上面的语句,执行结果是:

A 建表语句(1)报错

B 插入语句(2)报错

C 插入语句(3)报错

D 均不报错

414632a175d90c3f4b1d6edb01673d93.png

实操如上,所以答案是:【C 插入语句(3)报错】

建表语句可以不主动设置主键非空(上例中的id),InnoDB会自动添加非空(not null)且唯一(unique)的限制。

d735a7fe0ea87f9655c487fd06b75c23.png

使用show create table查看,会发现主键列要求非空,且默认值是0,因此:

insert into user(name) values('shenjian');

id默认为0,第一次插入成功,第二次插入主键冲突。

画外音:这一题很多人答错。

练习三:建表时,可不可以选择多个字段做主键?

(1) create table user(

id int not null,

name varchar(10) not null,

primary key(id, name)

)engine=innodb;

(2) insert into user values(1, 'shenjian');

(3) insert into user values(1, 'zhangsan');

(4) insert into user values(2, 'shenjian');

画外音:建表时,声明联合主键(a,b),插入若干元素,有些a重复,有些b重复。

提问,连续执行上面的语句,执行结果是:

A 建表语句(1)报错

B 插入语句(2)报错

C 插入语句(3)报错

D 插入语句(3)报错

E 均不报错

1859e3ac3bedd90a6fe5249c78ce27ff.png

实操如上,所以答案是:【E 均不报错】

这就是所谓的联合主键,不要求每个列唯一,而要求联合主键各个列的“组合唯一”。

画外音:联合索引,联合主键,类似的。

练习四:可不可以主动插入自增主键?

(1) create table user(

id int auto_increment,

name varchar(10) not null,

primary key(id)

)engine=innodb;

(2) insert into user(name) values('shenjian');

(3) insert into user(id, name) values(10,'shenjian');

(4) insert into user(name) values('shenjian');

画外音:建表时,自增ID为主键,插入若干元素,有些包含自增ID,有些不包含。

提问,连续执行上面的语句,执行结果是:

A 建表语句(1)报错

B 插入语句(2)报错

C 插入语句(3)报错

D 插入语句(3)报错

E 均不报错

a259dba60db1d2fd5c762f0ea98db505.png

实操如上,所以答案是:【E 均不报错】

自增ID经常被用来作为主键,插入行记录时,不指定值,InnoDB自动增加。同时,也可以手动指定值:

(1)手动指定的值与已有值重复,则主键冲突;

(2)与已有值不冲突,则插入成功;

(3)未来不指定值插入,会从手动插入值往后继续递增;

3f63af2b6a7d663e2fd22d58e86f0135.png

如上例,手动插入10后,未来插入的行会从11开始,自增ID中间出现空洞。

练习五:建表时,可不可以使用联合自增主键?

(1) create table user(

id int auto_increment,

name varchar(10) not null,

primary key(name, id)

)engine=innodb;

(2) insert into user(name) values('shenjian');

(3) insert into user(id, name) values(10,'shenjian');

(4) insert into user(name) values('shenjian');

画外音:建表时,声明联合主键(a,b),并且有一个是自增ID,插入若干元素,包含自增ID,有些不包含。

提问,连续执行上面的语句,执行结果是:

A 建表语句(1)报错

B 插入语句(2)报错

C 插入语句(3)报错

D 插入语句(3)报错

E 均不报错

8f9f05de74a774cda7d0c6ea3aad79fd.png

实操如上,所以答案是:【A 建表语句(1)报错】

这里不是指自增ID不能作为联合主键,而是如果作为联合主键,必须放在第一列。

3690c30476df6c1ffc24eb1d1ed412e8.png

primary key(name, id)

改为

primary key(id, name)

就可以执行成功了。

画外音:这一题很多人答错。

总结

一:InnoDB建表时,可不可以不声明主键?

可以不声明主键,但必须要有聚集索引:

(1)有主键,主键是聚集索引;

(2)没有主键,首个非空唯一列是聚集索引;

(3)没有符合条件的列,row-id是聚集索引;

主键和聚集索引不是一个东西,不要混淆。

二:InnoDB建表时,可不可以不声明主键非空?

可以不声明主键非空,会自动加上非空限制。

三:InnoDB建表时,可不可以选择多个字段做主键?

可以使用联合主键,组合列唯一即可。

四:InnoDB插入时,可不可以主动插入自增主键?

可以指定自增列的值,但可能导致空洞。

五:InnoDB建表时,可不可以使用联合自增主键?

可以,但自增ID必须在联合主键的第一列。

希望大家对主键有更系统性的认识。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值