问答

Mysql: nullable的列上加联合索引起作用吗?

作者:admin 2021-09-19 我要评论

我一张表中有两列,两个都是字符串,一个是字符串代表的日期startDay,格式诸如"2020-03-23",一个是字符串orderId,都是可null的 创建一个联合索引, 现在想要查 ...

在说正事之前,我要推荐一个福利:你还在原价购买阿里云、腾讯云、华为云服务器吗?那太亏啦!来这里,新购、升级、续费都打折,能够为您省60%的钱呢!2核4G企业级云服务器低至69元/年,点击进去看看吧>>>)

我一张表中有两列,两个都是字符串,一个是字符串代表的日期startDay,格式诸如"2020-03-23",一个是字符串orderId,都是可null的

创建一个联合索引, 现在想要查

  select * from table where startDay < '2020-03-23' and orderId is null
  

由于第一个列是范围索引,导致第二个列索引不可用,联合索引(startDay,orderId), 这样不行,因为startDay < '2020-03-23'的候选记录太多,这样性能很糟糕,所以想创建联合索引(orderId, startDay)

  select * from table where orderId is null and startDay < '2020-03-23'
  
  select * from table where orderId is not null and startDay < '2020-03-23'
  

但是不知道用这个(orderId,startDay)联合索引会让两个列都起作用嘛?

###

可以.
实验如下:

create table user_test(
    `id` bigint auto_increment COMMENT '主键ID',
    `birthday` varchar(1024) null COMMENT '出生日期',
    `name` varchar(1024) null COMMENT '姓名',
    `country` varchar(1024) not null COMMENT '国家',
    `city` varchar(1024) not null COMMENT '城市',
    PRIMARY KEY (`id`),
    KEY `IDX_BIRTHDAY_NAME` (`name`,`birthday`)
)ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='用户信息';

INSERT user_test (`birthday`,`name`,`country`,`city`) values ('1990-03-01','Ben','china','shanghai'),('1993-03-01','Tom','china','shanghai'),('1998-03-01',null,'china','shanghai'),('1990-03-01','Ben','china','shanghai'),('2004-03-01',null,'china','shanghai');

explain select * from user_test where birthday > '1995-03-01' and name is null;

image.png

可以看到命中了联合索引

虽然MySQL可以在含有null的列上使用索引,但不代表null和其他数据在索引中是一样的。不建议列上允许为空。最好限制not null,并设置一个默认值,比如0''空字符串等,如果是datetime类型,可以设置成'1970-01-01 00:00:00'这样的特殊值。对MySQL来说,null是一个特殊的值,Conceptually, NULL means “a missing unknown value” and it is treated somewhat differently from other values。比如:不能使用=,<,>这样的运算符,对null做算术运算的结果都是nullcount时不会包括null行等,null比空字符串需要更多的存储空间等。

参考文章:

版权声明:本文转载自网络,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。本站转载出于传播更多优秀技术知识之目的,如有侵权请联系QQ/微信:153890879删除

相关文章
  • Mysql: nullable的列上加联合索引起作

    Mysql: nullable的列上加联合索引起作

  • 变量声明,有的情形声明为“空”,又有

    变量声明,有的情形声明为“空”,又有

  • 对象 {a:{b:{c:{}}}}}如何转换成a.b.c

    对象 {a:{b:{c:{}}}}}如何转换成a.b.c

  • 如何理解Object.prototype.toString.ca

    如何理解Object.prototype.toString.ca

腾讯云代理商
海外云服务器