在外键上加索引与没有索引的区别:

主要有两方面问题:一、在DML操作时的阻塞问题,二 、DML操作时的速度问题

一、阻塞问题

外键在无索引的情况下,更新主表外键关联字段时,需要为子表加4级锁(S);在有索引的情况下,更新主表外键关联字段时,需要为子表加2级锁(RS)。在子表上本来就已有RX时,S锁无法被兼容,造成更新主表阻塞。如果子表上本来没有锁,更新主表的操作不被阻塞时(更新完后我们暂时不commit),此刻,如果外键没有索引,4级锁(S)是“瞬间”加上的,然后就释放不易观察到;而在有索引的情况下,给子表加的2级锁(RS)会一直存在,直到更新主表的会话回滚或提交。

无论在有无外键索引的情况下,子表插入数据,需要给主表加2级锁(RS),这个操作是否会被阻塞,要看主表中对应的记录是否存在RX锁。这里是用delete进行的测试,其它DML语句情况相同。

1. 创建两张表并插入数据,模拟实验环境

BALLONTT@PROD> create table dept(deptno number,dname varchar2(10));

BALLONTT@PROD> alter table dept add constraint pk_dept primary key(deptno);

BALLONTT@PROD> create table emp(empno number,ename varchar2(10),deptno number);

BALLONTT@PROD> alter table emp add constraint fk_emp foreign key(deptno) references dept(deptno);

BALLONTT@PROD> insert into dept values(01,'aa');

BALLONTT@PROD> insert into dept values(02,'bb');

BALLONTT@PROD> insert into dept values(03,'cc');

BALLONTT@PROD> insert into dept values(04,'dd');

BALLONTT@PROD> commit;

Commit complete.

BALLONTT@PROD> insert into emp(empno,deptno) values(111,01);

BALLONTT@PROD> insert into emp(empno,deptno) values(222,02);

BALLONTT@PROD> commit;

Commit complete.

2. 确认表的信息

BALLONTT@PROD> select * from dept;

DEPTNO DNAME

---------- ----------

1 aa

2 bb

3 cc

4 dd

BALLONTT@PROD> select empno,deptno from emp;

EMPNO     DEPTNO

---------- ----------

111          1

222          2

BALLONTT@PROD> select a.object_id,a.object_name,l.session_id from

2  all_objects a,v$locked_object l

3  where a.object_id=l.object_id;

no rows selected

3. 在会话1(session_id=125)中执行下面DML操作(此时emp表中没有索引时)

BALLONTT@PROD> insert into emp(empno,deptno) values(333,3);

1 row created.

查看被锁的对象信息

BALLONTT@PROD> select a.object_id,a.object_name,l.session_id from

2  all_objects a,v$locked_object l

3  where a.object_id=l.object_id;

OBJECT_ID     OBJECT_NAME                    SESSION_ID

-----------  ---------------------        ----------------

9752   EMP                               125

9750   DEPT                              125

BALLONTT@PROD> select sid,type,id1,id2,lmode,request from v$lock where sid=125;

SID TY        ID1        ID2        LMODE    REQUEST

---------- --    ----------     ------     ----      ----------

125 TM       9750          0          2          0

125 TM       9752          0          3          0

125 TX      65558        105          6          0

上面对emp的插入操作,对dept(id:9750)加2模式表级锁(即RS锁),对EMP(id:9752)加表级锁RX(LMODE 3),和行级锁X(LMODE 6)

4. 紧着着在会话2(session_id=113)中对主键所在表进行DML,查看是否阻塞

BALLONTT@PROD> update dept set deptno=10 where deptno=3;---阻塞

查看锁的信息

BALLONTT@PROD> select sid,type,id1,id2,lmode,request from v$lock where sid in(113,125);

SID TY        ID1        ID2      LMODE    REQUEST

---------- -- ---------- ---------- ---------- ----------

125 TM       9750          0          2          0

125 TM       9752          0          3          0

113 TM       9750          0          3          0

113 TM       9752          0          0          4

125 TX      65558        105          6          0

(9750代表dept,9752代表emp)对dept的更新需要在表dept上加表级锁RX,同时向EMP表申请S锁(REQUEST 4)。但由于此时EMP上有插入操作带来的RX锁,与S锁不兼容,所以因无法得到S锁而导致对DEPT的更新操作阻塞。

update dept set deptno=16 where deptno=4; --同阻塞,原因如上。

5. 终止会话2,回滚会话1,在EMP表的外键上加索引

BALLONTT@PROD> create index ind_emp on emp(deptno);

6.重复上面的3步骤,并在会话2中在执行下面语句(有外键索引,下面语句需要先对子表加RS锁,然后再去申请主表RX锁,是否会被阻塞取决于子表需要被加RS锁的记录上是否已有RX锁,主表需要被加RX锁的记录是否有RS锁)

BALLONTT@PROD> update dept set deptno=10 where deptno=4;--不阻塞

查看锁的的信息:

BALLONTT@PROD> select sid,type,id1,id2,lmode,request from v$lock where sid in(113,125);

SID TY        ID1        ID2      LMODE    REQUEST

---------- -- ---------- ---------- ---------- ----------

125 TM       9750          0          2          0

125 TM       9752          0          3          0

113 TM       9750          0          3          0

113 TM       9752          0          2          0

113 TX     327726        105          6          0

125 TX     262156        107          6          0

6 rows selected.

会话1(sid:125)中对emp的插入操作形成了3个锁。(上文已说明)

会话2(sid:113)中对dept的更新操作也有三个锁,分别是在dept表上的常规更新带来的两个锁RX,和TX。第三锁为加在子表EMP上的RS锁。RS锁与EMP上已有的RX锁兼容,所以不会阻塞。

update dept set deptno=16 where deptno=3; --阻塞

二、DML操作时的速度问题·

当使用ON DELETE CASCADE删除父表中的记录时,如果在子表中的外键没有使用索引则当执行该操作时会对子表进行全表的扫描,而事实上这个全表的扫描是不需要的。更坏的情况是,如果删除多个父表中的记录,每删除一条记录则会进行一次全表扫描,可想而知,对于性能的影响是多么的大!

对于父表和子表的连接查询,情况也是类似的。当进行这种连接查询时,如果不对外键使用索引则会发现查询的速度大大降低。

由此可知,我们应该在外键上建立索引。

ballontt
2014/03/26

---The End---
微博:weibo.com/ballontt
如需转载,请标明出处和链接,谢谢!

【fk_index】外键中有无索引的区别相关推荐

  1. 单图说TDSQL;OceanBase 2.2 事务引擎核心功能;穿云箭2.0版发布;RMAN DUPLICATE配置19C DG;外键上有无索引的影响;MySQL8.0 索引新功能;GaussDB C

    摘要:墨天轮数据库周刊第16期发布啦,每周1次推送本周数据库相关热门资讯.精选文章.干货文档. 热门资讯 1.英国电信企业Virgin Media营销数据库配置错误 导致90万客户信息泄露 [摘要]英 ...

  2. oracle中主键和唯一索引的区别说明

    上周六去参加上海Oracle WDP俱乐部沙龙的一次交流,其中提到了关于"Oracle中主键和唯一索引的区别",基本上大家都可以说上几个,在网上也可以找到,但是总感觉不太全,根据自 ...

  3. Oracle外键需要建索引吗?

    关于Oracle中的外键,首先要说明一下. 1. 除非已定义了父表主键或唯一键约束,否则oracle将不允许创建子表的外键约束. 2. 在定义外键约束时,oracle不会自动创建索引,所以必须手动在与 ...

  4. 在外键上创建索引V1.0

    开始: 发现在一个项目中,后台数据库各个表,具有外键约束的列上都没有创建索引.我们需要一次性给他们创建索引,下面我写了一个通用的存储过程sp_CreateIndexForFK,来实现这一个功能. 也许 ...

  5. 数据库主键、外键和唯一键的区别

    数据库主键.外键和唯一键的区别 现在我们想建立学生表,用来存储,一个系统的登陆信息.建表如下: create table student4 (stu_id int primary key identi ...

  6. SQL Server外键中的DELETE CASCADE和UPDATE CASCADE

    In this article, we will review on DELETE CASCADE AND UPDATE CASCADE rules in SQL Server foreign key ...

  7. mysql学习笔记 ----外键.myisam和innodb的区别

    外键:InnoDB支持外键. 若要设置外键,则参照表和被参照表都必须对相应字段设置索引.设置完索引便可 设置外键.可以在定义外键的时候,在最后加入这样的关键字: ON UPDATE CASCADE; ...

  8. 主键和唯一索引的区别

    原文地址:http://blog.csdn.net/baoqiangwang/article/details/4832814 主键是一种约束,唯一索引是一种索引,两者在本质上是不同的. 区别 1. 主 ...

  9. 主键和唯一性索引的区别

    主键一定是唯一性索引,唯一性索引并不一定就是主键.  所谓主键就是能够唯一标识表中某一行的属性或属性组,一个表只能有一个主键,但可以有多个候选索引. 因为主键可以唯一标识某一行记录,所以可以确保执行数 ...

  10. mysql主键和唯一索引_mysql 主键和唯一索引的区别

    主键是一种约束,唯一索引是一种索引,两者在本质上是不同的. 主键创建后一定包含一个唯一性索引,唯一性索引并不一定就是主键. 唯一性索引列允许空值,而主键列不允许为空值. 主键列在创建时,已经默认为非空 ...

最新文章

  1. docker初体验:docker部署nginx服务
  2. 安装oel6.3 操作系统
  3. 远程桌面复制文件,由于网络或其他原因被意外中断,后来再连上远程桌面就无法复制了,而且复制文件的对话框也无法取消,可以试试下面的方法,实测有效:
  4. 四种引用类型(强引用、软引用、弱引用、虚引用)的简单介绍
  5. Kotlin入门(16)容器的遍历方式
  6. 建立一个中文名字的文件夹Python
  7. AI团队领头人携耗资数十万项目案例,价值2W+的5堂直播课,限时免费Get!
  8. mysql 存储过程 锁表_MYSQL锁表问题的解决方法
  9. 华为版计算机软件,Huawei华为手机PC客户端软件
  10. CleanMyMac X2022纯净版苹果电脑管家
  11. QQ农场启示录之:偷菜成本核算
  12. 企业要如何利用360评估法做好人才盘点?
  13. 【MongoDB-MongoVUE图像管理工具】
  14. 解决IOS微信浏览器底部会出现向前向后返回按钮,返回不刷新的问题
  15. 等额本金VS等额本息
  16. UVa 1600 巡逻机器人(Patrol Robot)
  17. 如何在react中禁止浏览器后退
  18. 给站长与网络推广人员的文章
  19. 2020笔记本性价比之王_2020十大笔记本电脑性价比排行(最新笔记本电脑推荐)...
  20. 根据Spring上下文对象xxxContext获得*.properties属性文件中的配置属性值

热门文章

  1. 超实用的自我规划模型 | 进击
  2. Electron入门宝典(三)菜单快捷键
  3. 【printf】打印数值、打印二进制、打印十六进制、sprintf
  4. variance和variation的区别
  5. signature=4a882a48c4a4b2b41835e11b6fafa69f,ABB 38SC980002R375
  6. Java常用类--java.lang.StringBuilder
  7. IP和子网掩码和网关的关系
  8. 惠普Elite Mini 800 G9 评测
  9. rstudio 连接mysql_RStudio 操作MySQL数据库
  10. Pytorch下基于lstm的股价预测