广告
返回顶部
首页 > 资讯 > 数据库 >类型隐式转换导致的?No,并不是
  • 198
分享到

类型隐式转换导致的?No,并不是

类型隐式转换导致的?No,并不是 2017-06-27 08:06:30 198人浏览 绘本
摘要

本文作者:叶金荣,知数堂联合创始人,Mysql DBA课程讲师。oracle mysql ACE,Mysql布道师。有多年MySQL及系统架构设计经验,擅长MySQL企业级应用、数据库设计、优化、故障处理等。 疑似类型隐式转换一例

类型隐式转换导致的?No,并不是

本文作者:叶金荣,知数堂联合创始人,Mysql DBA课程讲师。oracle mysql ACE,Mysql布道师。有多年MySQL及系统架构设计经验,擅长MySQL企业级应用、数据库设计、优化、故障处理等。

疑似类型隐式转换一例

有群友提了下面这样的问题

请教个隐式转换的问题:
SELECT count(*) FROM test WHERE time >= 2019-05-17;
time列是datetime类型,
这条SQL的执行结果是相当于 where 1, 这个是什么原因呢?

SQL执行有个warnings:
Warning | 1292 | Incorrect datetime value: "1997" for column "time" at row

从告警信息来看,是把 2019-05-17 做了数学减法运算,得到常量 1997, 
再把常量1997转换为 datetime 类型,再跟time字段做比较。

但用函数 cast(1997 as datetime),也是同样的告警信息,但结果是 NULL

那么该SQL是否可以等价为:
SELECT count(*) FROM test WHERE time >= NULL

这个SQL结果会是0,因为跟NULL值比较的结果是NULL。

虽然WHERE条件错写成一个算式,但执行时没有报错,只有一个告警信息,

感觉还是因为发生了类型隐式转换,用不到索引,否则不会是全表扫描。

这是我的第一次回复内容

事实上,条件 WHERE time >= 2019-05-17,
的意思是:time >= 1997,这是表达式 2019-05-17 的结算结果。
这个不是类型隐式转换,是你SQL没写对。

我们看到SQL的执行计划是这样的

对于第二个疑问:为什么会走全表扫描计划呢?

我的看法是这样的:首先,上面的SQL条件相当于 WHERE time >= 1997。其次,MySQL认为"1997"不是合法的日期时间类型数据,看到执行计划中有告警

Incorrect datetime value: "1997" for column

因此,time >= 1997 这个条件,就会被当做一个逻辑表达式,因为不是 0(FALSE),也不是 NULL,所以就会被认为是永远为真(TRUE)。也就是说,time列中所有不是FALSE或NULL的值都符合条件。

我们可以测试确认这个说法:

# 表中dt列是datetime类型,但允许为NULL
[root@yejr.me]> show create table t1G
*************************** 1. row ***************************
       Table: t1
Create Table: CREATE TABLE `t1` (
  `c1` int NOT NULL,
...
  `dt` timestamp NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
  PRIMARY KEY (`c1`),
  KEY `k2` (`dt`)
) ENGINE=InnoDB;


# 查看所有数据
[root@yejr.me]> select * from t1;
+-----+-----+---------------------+
| c1  | ... | dt                  |
+-----+-----+---------------------+
|   2 | ... | NULL                |
|   3 | ... | 2017-11-01 15:44:27 |
|   5 | ... | 2020-02-13 16:02:55 |
+-----+-----+---------------------+
3 rows in set (0.00 sec)


# c1=2的记录中,dt列值为NULL,不符合条件
[root@yejr.me]> select * from t1 where dt >= 1997;
+-----+-----+---------------------+
| c1  | ... | dt                  |
+-----+-----+---------------------+
|   3 | ... | 2017-11-01 15:44:27 |
|   5 | ... | 2020-02-13 16:02:55 |
+-----+-----+---------------------+
2 rows in set (0.00 sec)

很明显,只要表中dt列值不为NULL、不为0(符合日期时间格式的数据也肯定不会是0)的数据都会被读取到。

这种情况下,即便dt列有索引,也会因为需要扫描的数据太多,从而优化器认为直接走全表扫描的效率要更好,所以也无法使用索引。

还有个疑问,WHERE条件写成 time >= cast(1997 as datetime) 时会怎样?这种情况下,因为 cast(1997 as datetime) 的结果是 NULL,所以WHERE条件等同于time>= NULL,对NULL的运算是不能这么写的,而应该写成 dt IS NULLt IS NOT NULL才对。所以,这么写的话,这个查询是不会有任何结果的,包括列值为NULL的数据。

最后的小结:

写SQL时,WHERE条件值记得总是带上引号,避免发生意想不到的情况。

对NULL值的判断,必须是用 IS NULL 或 IS NOT NULL,不能是大小值的判断。另外,WHERE条件中的NULL其实是可以用到索引的,例如:

[root@yejr.me]> desc select * from t1 where dt is NULLG
************************ 1. row ************************
           id: 1
  select_type: SIMPLE
        table: t1
   partitions: NULL
         type: ref
possible_keys: k2
          key: k2
      key_len: 5
          ref: const
         rows: 1
     filtered: 100.00
        Extra: Using where

除了防范类型隐式转换,还要注意防范字符集隐式转换,具体参考MySQL手册12.2 Type Conversion in Expression Evaluation

您可能感兴趣的文档:

--结束END--

本文标题: 类型隐式转换导致的?No,并不是

本文链接: https://www.lsjlt.com/news/5143.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 类型隐式转换导致的?No,并不是
    本文作者:叶金荣,知数堂联合创始人,MySQL DBA课程讲师。Oracle MySQL ACE,MySQL布道师。有多年MySQL及系统架构设计经验,擅长MySQL企业级应用、数据库设计、优化、故障处理等。 疑似类型隐式转换一例 ...
    99+
    2017-06-27
    类型隐式转换导致的?No,并不是
  • MySQL隐式类型转换导致索引失效
    今天发现一个问题,where条件的列上明明有索引,但是执行计划还是走全表扫描mysql>  explain select task_id&n...
    99+
    2022-10-18
  • MySQL隐式类型转换导致索引失效的解决
    目录问题 复现 隐式转换 总结 参考 问题 在工作中发现,有一个接口只执行一条SQL查询语句,并且SQL明明使用了主键列,但是速度很慢。 在MySQL中EXPLAINN后发现,执行...
    99+
    2022-11-12
  • Oracle因数据不一致而导致的隐式转换错误一例
       今天,开发同事说他在测试库执行一条SQL的时候,报ORA-01722: invalid number错误,但是在生产库和灰度库执行同一条SQL却能够正常执行,SQL如下: &nbs...
    99+
    2022-10-18
  • go为什么不支持隐式类型转换
    go不支持隐式类型转换是因为它会降低代码可读性和可维护性、增加潜在的错误风险、破坏语言的一致性和简洁性、鼓励明确的类型转换和显式的编程风格等。详细介绍:1、降低代码可读性和可维护性,隐式类型转换在代码中往往不清晰,难以阅读和理解,当程序员在...
    99+
    2023-11-10
    隐式类型转换 Go
  • C语言隐式类型转换与强制类型转换的方法是什么
    本篇内容主要讲解“C语言隐式类型转换与强制类型转换的方法是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“C语言隐式类型转换与强制类型转换的方法是什么”吧!类型转换数据有不同的类型,不同类型数...
    99+
    2023-06-25
  • 浅析JavaScript中的隐式类型转换
    目录为什么会出现隐式类型转换隐式类型转换的内部转换机制何时会触发ToPrimitive或ToNumber具体案例分析总结为什么会出现隐式类型转换 这个问题的本质原因是因为JavaSc...
    99+
    2023-03-08
    JavaScript隐式类型转换 JavaScript隐式类型
  • JavaScript中的隐式类型如何转换
    这篇文章主要介绍“JavaScript中的隐式类型如何转换”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“JavaScript中的隐式类型如何转换”文章能帮助大家解决问题。为什么会出现隐式类型转换这个...
    99+
    2023-07-05
  • JavaScript中类型的强制转换与隐式转换详解
    目录一、隐式转换双等号里的转换 Boolean 类型转换 "+" 与 "-" 二、强制类型转换 new String 与 ' ' 总结一、隐式转换 以下语句的执行结果是什么? A...
    99+
    2022-11-12
  • 深入探讨PHP类型的隐式转换
    PHP是一种广泛使用的服务器端编程语言,它支持多种数据类型。PHP的数据类型有两种:基本数据类型和复合数据类型。在PHP中,数据类型的转换通常是隐式的,这是由PHP的动态类型语言特性所决定的。但是,隐式类型转换可能会引起一些问题和错误,这篇...
    99+
    2023-05-14
    php
  • PHP类型的隐式转换怎么实现
    本篇内容主要讲解“PHP类型的隐式转换怎么实现”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“PHP类型的隐式转换怎么实现”吧!一、基本数据类型在PHP中,基本数据类型有四种,分别是整型(int)...
    99+
    2023-07-05
  • 不同数据类型之间的隐含转换
    赋值语句中数据类型之间的隐含转换    将变长字符(varchar2)或定长字符型(char)转换成数字型(NUMBER)   &nbs...
    99+
    2022-10-18
  • c++隐式类型转换存在的问题解析
    目录什么是隐式转换:为什么要进行隐式转换:隐式转换的原则:目标代码构造函数定义的隐式类型转换分析a1分析a2分析a3什么是隐式转换: c++中的基本类型并非完全对立,部分类型之间是可...
    99+
    2022-11-13
  • JavaScript运算符规则与隐式类型的转换
    本篇内容介绍了“JavaScript运算符规则与隐式类型的转换”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成...
    99+
    2022-10-19
  • SQL Server 中的数据类型隐式转换问题
    写这篇文章的时候,还真不知道如何取名,也不知道这个该如何将其归类。这个是同事遇到的一个案例,案例比较复杂,这里抽丝剥茧,仅仅构造一个简单的案例来展现一下这个问题。我们先构造测试数据,如下所示: CRE...
    99+
    2022-10-18
  • JavaScrip简单数据类型隐式转换的实现
    目录JavaScrip隐式转换规则JavaScrip类型转换详解转换成Number类型转换为String类型转换为Boolean类型JavaScrip特殊操作符对类型转换的影响!逻辑...
    99+
    2023-05-20
    JavaScrip 数据类型隐式转换 JavaScrip  隐式转换
  • MySql类型转换导致行锁升级为表锁的示例
    小编给大家分享一下MySql类型转换导致行锁升级为表锁的示例,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!在MySql的写语句中,给表列赋值与表类型不符合时,MySql底层的优化器发挥作用...
    99+
    2022-10-18
  • Mysql 5.6 "隐式转换"导致的索引失效和数据不准确的问题
    背景 在一次进行SQl查询时,我试着对where条件中vachar类型的字段去掉单引号查询,这个时候发现这条本应该很快的语句竟然很慢。这个varchar字段有一个复合索引。其中的总条数有58989,甚...
    99+
    2022-05-17
    Mysql 5.6隐式转换导致的索引失效 Mysql 5.6隐式转换
  • 直观详细的typescript隐式类型转换图文详解
    正文 1、unknown是所有类型的父类型,其他类型都可以赋值给 unknown let a: undefined = undefined; let b: null = null;...
    99+
    2022-11-13
  • Mysql 5.6 "隐式转换"导致的索引失效和数据不准确的解决方法
    这篇文章主要介绍了Mysql 5.6 "隐式转换"导致的索引失效和数据不准确的解决方法,具有一定借鉴价值,需要的朋友可以参考下。希望大家阅读完这篇文章后大有收获。下面让小编带着大家一起了...
    99+
    2022-10-18
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作