设为首页 - 加入收藏 延边站长网 (http://www.0433zz.com)- 国内知名站长资讯网站,提供最新最全的站长资讯,创业经验,网站建设等!
热搜: mini 系统 为什 为啥
当前位置: 首页 > 站长学院 > MySql教程 > 正文

MySQL中的主键和rowid,看似简单,其实有一些使用陷阱需要注意

发布时间:2019-10-17 12:11 所属栏目:[MySql教程] 来源:杨建荣
导读:大家在MySQL中我们可能听到过rowid的概念,但是却很难去测试实践,不可避免会有一些疑惑,比如: 1)如何感受到rowid的存在 2)rowid和主键有什么关联关系 3)在主键的使用中存在哪些隐患 4)如何来理解rowid的潜在瓶颈并调试验证 今天要和大家一起讨论这

MySQL中的主键和rowid,看似简单,其实有一些使用陷阱需要注意

大家在MySQL中我们可能听到过rowid的概念,但是却很难去测试实践,不可避免会有一些疑惑,比如:

1)如何感受到rowid的存在

2)rowid和主键有什么关联关系

3)在主键的使用中存在哪些隐患

4)如何来理解rowid的潜在瓶颈并调试验证

今天要和大家一起讨论这几个问题,测试的环境基于MySQL 5.7.19版本。

问题1:如何感受到rowid的存在

我们不妨通过一个案例来进行说明。

记得有一天统计备份数据的时候,写了一条SQL,当看到执行结果时才发现SQL语句没有写完整,在完成统计工作之后,我准备分析下这条SQL语句。

  1. mysql>?select?backup_date?,count(*)?piece_no?from?redis_backup_result;??
  2. +-------------+----------+??
  3. |?backup_date?|?piece_no?|??
  4. +-------------+----------+??
  5. |?2018-08-14?|?40906?|??
  6. +-------------+----------+??
  7. 1?row?in?set?(0.03?sec)?

根据业务特点,一天之内肯定没有这么多的记录,明显不对,到底是哪里出了问题呢。

自己仔细看了下SQL,发现是没有加group by,我们随机查出10条数据。

  1. mysql>?select?backup_date?from?redis_backup_result?limit?10;??
  2. +-------------+??
  3. |?backup_date?|??
  4. +-------------+??
  5. |?2018-08-14?|??
  6. |?2018-08-14?|??
  7. |?2018-08-14?|??
  8. |?2018-08-15?|??
  9. |?2018-08-15?|??
  10. |?2018-08-15?|??
  11. |?2018-08-15?|??
  12. |?2018-08-15?|??
  13. |?2018-08-15?|??
  14. |?2018-08-15?|??
  15. +-------------+??
  16. 10?rows?in?set?(0.00?sec)?

在早期的版本中数据库参数sql_mode默认为空,不会校验这个部分,从语法角度来说,是允许的;但是到了高版本,比如5.7版本之后是不支持的,所以解决方案很简单,在添加group by之后,结果就符合预期了。

  1. mysql>?select?backup_date?,count(*)?piece_no?from?redis_backup_result?group?by?backup_date;??
  2. +-------------+----------+??
  3. |?backup_date?|?piece_no?|??
  4. +-------------+----------+??
  5. |?2018-08-14?|?3?|??
  6. |?2018-08-15?|?121?|??
  7. |?2018-08-16?|?184?|??
  8. |?2018-08-17?|?3284?|??
  9. |?2018-08-18?|?7272?|??
  10. |?2018-08-19?|?7272?|??
  11. |?2018-08-20?|?7272?|??
  12. |?2018-08-21?|?7272?|??
  13. |?2018-08-22?|?8226?|??
  14. +-------------+----------+??
  15. 9?rows?in?set?(0.06?sec)?

但是比较好奇这个解析的逻辑,看起来是SQL解析了第一行,然后输出了count(*)的操作,显然这是从执行计划中无法得到的信息。

我们换个思路,可以看到这个表有4万多条的记录。

  1. mysql>?select?count(*)from?redis_backup_result;??
  2. +----------+??
  3. |?count(*)?|??
  4. +----------+??
  5. |?40944?|??
  6. +----------+??
  7. 1?row?in?set?(0.01?sec)?

为了验证,我们可以使用_rowid的方式来做初步的验证。

InnoDB表中在没有默认主键的情况下会生成一个6字节空间的自动增长主键,可以用select _rowid from table来查询,如下:

  1. mysql>?select?_rowid?from?redis_backup_result?limit?5;??
  2. +--------+??
  3. |?_rowid?|??
  4. +--------+??
  5. |?117?|??
  6. |?118?|??
  7. |?119?|??
  8. |?120?|??
  9. |?121?|??
  10. +--------+??
  11. 5?rows?in?set?(0.00?sec)?

【免责声明】本站内容转载自互联网,其相关言论仅代表作者个人观点绝非权威,不代表本站立场。如您发现内容存在版权问题,请提交相关链接至邮箱:bqsm@foxmail.com,我们将及时予以处理。

网友评论
推荐文章