count(字段) 、count(主键 id)、 count(1)、count(*)

转自 :14 | count(*)这么慢,我该怎么办? 丁奇老师的极客时间专栏 mysql45讲。

1- 对于 count(字段) 来说
如果这个“字段”是定义为 not null 的话,一行行地从记录里面读出这个字段,判断不能为 null,按行累加;
如果这个“字段”定义允许为 null,那么执行的时候,判断到有可能是 null,还要把值取出来再判断一下,不是 null 才累加。
 
2- 对于 count(主键 id) 来说,InnoDB 引擎会遍历整张表,把每一行的 id 值都取出来,返回给 server 层。server 层拿到 id 后,判断是不可能为空的,就按行累加。
 
3- 对于 count(1) 来说,InnoDB 引擎遍历整张表,但不取值。server 层对于返回的每一行,放一个数字“1”进去,判断是不可能为空的,按行累加。
 
4- 但是 count(*) 是例外,并不会把全部字段取出来,而是专门做了优化,不取值。count(*) 肯定不是 null,按行累加。
 
count(字段)<count(主键 id)<count(1)≈count(*),所以我建议你,尽量使用 count(*)。
 
补充:由于Mysql事务以及MVCC(多版本并发控制)  的原因,该专栏 03 | 事务隔离:为什么你改了我还看不见? 以及  08 | 事务到底是隔离的还是不隔离的? 讲解的很详细
在取每行的数据的时候会去判断这行数据对于当前事务来说是否是可见的(mysql事务默认隔离级别为可重复读),当表中数据很大,这些判断会很耗时。
所以业务中需要频繁查询表中记录条数且后期表中数据量很大的话, 最好还是单独存起来 ,应用缓存或者Redis都可以。
 

原文地址:https://www.cnblogs.com/stephenzengx/p/13692383.html