sql server 字符串转数字带默认值 sql把字符串转为数字_SQL


“数据库里明明没有符合条件的数据,为什么结果还是有数据返回”,最近好几个同事差一点被这个问题难住了。

事情是这样的,前几天,一个在生产环境运行了好几个月的功能在测试环境调试发生了一个异常,也就是数据库里明明没有某个用户的数据,但在实际查询时却返回了一些无关数据,导致一个本来应该判断为空的逻辑,结果判断为不为空。

定位了好久,后来终于找到了问题所在。原来,就是下面这个SQL查询语句出现了问题。


sql server 字符串转数字带默认值 sql把字符串转为数字_浮点数_02


结合上面的SQL语句和下面的表结构,大家也可以思考思考,如果让你来执行这个SQL,你能预知结果吗?为什么?


sql server 字符串转数字带默认值 sql把字符串转为数字_字符串转数字 sql_03


相信很多人会认为,这结果就是空,但实际呢?


sql server 字符串转数字带默认值 sql把字符串转为数字_浮点数_04


实际上,如上图所示,结果不为空。

要解释这样的结果,那就不得不说MySQL中数据类型的隐式转换。

在MySQL中,当操作符与不同类型的操作数一起使用时,会发生类型隐式转换以使操作数兼容。在MySQL中,数据类型隐式转换的规则主要有以下几条:

  1. 两个参数至少有一个是 NULL 时,比较的结果也是 NULL,例外是使用 <=> 对两个 NULL 做比较时会返回 1,这两种情况都不需要做类型转换
  2. 两个参数都是字符串,会按照字符串来比较,不做类型转换
  3. 两个参数都是整数,按照整数来比较,不做类型转换
  4. 十六进制的值和非数字做比较时,会被当做二进制串
  5. 有一个参数是 TIMESTAMP 或 DATETIME,并且另外一个参数是常量,常量会被转换为 timestamp
  6. 有一个参数是 decimal 类型,如果另外一个参数是 decimal 或者整数,会将整数转换为 decimal 后进行比较,如果另外一个参数是浮点数,则会把 decimal 转换为浮点数进行比较
  7. 所有其他情况下,两个参数都会被转换为浮点数再进行比较

前面提到的执行异常的SQL查询语句,满足第7条,也就是name和false会被先转换成浮点数,然后再做比较。

在Mysql中,以数字开头的字符串,转换成浮点数时,会直接使用开头的数字;而不是以数字开头的字符串,转换成浮点数时,会直接变成0。如下:


sql server 字符串转数字带默认值 sql把字符串转为数字_SQL_05


至于bool值,false转换成浮点型会变成0,true转换成浮点型会变成1。如下:


sql server 字符串转数字带默认值 sql把字符串转为数字_字符串转数字 sql_06


这样一来,原来的SQL语句,在实际执行时就变成了下面这样:


sql server 字符串转数字带默认值 sql把字符串转为数字_SQL_07


如此一来,当然就可以查出数据来呀。唉,真是“一文钱难倒英雄汉”。

以后写SQL查询语句时,一定要确保查询条件的类型与数据库对应字段类型是一致的。