MySQL 查询栽坑笔记

前言

这几天在做数据分析的时候,数据要自己从数据库里面拿(之前都是别人把数据库导出来给我的说),心塞。还好之前学的SQL没有忘掉,咱连上数据库就开始写查询语句。之后就栽坑里了,执行的结果与预期的结果相差甚远,哭~。下面我把我栽坑的原因写下来,以帮助到可能会在此栽坑的开发人员。(PS: SQL绝对不是我喜欢的编程语言)

正文

SQL作为一门的声明式语言,它的运行方式并不同于我们所熟知的命令式程序语言。

在此废话几句,解释下什么是声明式语言,什么是命令式语言。

声明式语言:告诉计算机我要什么结果,至于怎么做,那是它的事情。

命令式语言:告诉计算机怎么做,它会返回按照我要求做之后的结果。

命令式 VS 声明式.png

执行顺序的坑

按照上面的解释,大概你也会了解一些为什么有些SQL编程出来的东西和你实际想要的结果是有些出入的,因为我们在潜意识中的行为动作是按照命令式编程的思维方式思考的。就好像用户在告诉计算机:“先执行第一步,再执行第二步,并在执行第二步之前先检查一下是否满足条件A 和条件B ”。例如,我们有时候会在SQL中用变量传参、使用循环语句、调用函数等等,会按照命令式编程的思维编写。然而,敲黑板,SQL恰恰没有按照这种命令式编程的思维方式来处理问题。另外还有一点,SQL的执行顺序并不参照语法顺序。下面表格【1】能帮你理解到这一点。

执行命令 编写顺序 执行顺序
SELECT 1 8
DISTINCT(等) 2 9
FROM 3 1
JOIN 4 3
ON 5 2
WHERE 6 4
GROUP BY 7 5
WITH 8 6
HAVING 9 7
ORDER BY 10 10
LIMIT 11 11
... ... ...

也就是说在一个SQL语句开始执行的时候,最先执行的总是FROM操作,最后执行的是LIMIT操作,在执行过程中,每一个操作都会产生一张虚拟的表,每一个虚拟的表都会作为下一步处理的输入,直至最后一步操作前,这些虚拟的表对于用户都是透明的,最后一步的操作结果产生的表作为最终的输出返回给用户。

执行时使用的变量也有坑

在SQL中,用户不只是可以使用系统中提供的变量,用户也可以自定义一些变量以供快速开发使用。

对于一些语句中,比如SELECT,在使用用户自定义变量时会得到期望的效果,但是,这个是很不稳定的。比如下面的语句,按照命令式编程思想地会认为MySQL会在第一轮和第二轮查询的时候,会给@b附上值,之后@b显示的是@a前两次的数值,但是事实上,@b会一直为0(不同的SQL解释器的运行机制不同,结果可能不同):

SET @a = 0;
SET @b = 0;
SELECT
    @b,
    @b=@a, 
    @a:= @a + 1 
FROM 
    sys.sys_config;

除此之外,还有另一个问题。变量的默认返回类型由语句开始时的类型决定的,正如下面的例子:

SET @a='test';
SELECT @a,(@a:=20) FROM tbl_name;

上述的SELECT语句中,MySQL会报告给客户端第一列的字段类型为字符串,同时将所有对@a变量的使用均转换为字符串处理,尽管在SELECT语句中将@a变量设置为数字类型。在SELECT语句执行后,@a变量才会在下一个语句中识别为数字类型。为了避免上述问题的发生,要么不在同一个语句中同时赋值并使用变量,要么在使用之前,将变量设置为0,0.0或者'',以确定它的数据类型。【2】

原文中有这样的一句话

In a SELECT statement, each select expression is evaluated only when sent to the client. This means that in a HAVING, GROUP BY, or ORDER BY clause, referring to a variable that is assigned a value in the select expression list does not work as expected.

在SELECT语句中,每个选择表达式仅在发送给客户端时才被计算。 这意味着在HAVING,GROUP BY或ORDER BY子句中,引用在选择表达式列表中指定值的用户自定义变量不能按预期工作。 也就是说用户自定义变量的值是在结果集发送到客户端后才计算的

这里有一种解释是:“MySQL优化器在某些场景下可能会将这些变量优化掉,这可能导致代码不按预想的方式运行。”

后记

关于用户自定义变量,如果运用的好,能够写出高效简洁的SQL语句,如果运用不当,也可能把自己给坑了(比如我? 哭~)。这个完全取决于使用它的人。

参考文档

【1】: MySQL技术内幕:SQL编程-第三章查询处理

【2】:译自 MySQL :: MySQL 5.7 Reference Manual :: 9.4 User-Defined Variables

如果本文帮到了你,那就点赞或者打个赏吧,我才不会谢谢你,哼~

By 土豆豆,2018年8月2日

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容