APP推广合作
联系“鸟哥笔记小乔”
sql数据运营(写给运营同学和初学者的 SQL 入门教程)
2024-01-10 08:44:10

写给运营同学和初学者的 SQL 入门教程

sql数据运营(写给运营同学和初学者的 SQL 入门教程)
  • Preface
  • Introduction
  • 1 预备知识
  • 1.1 数据库和数据表
  • 1.2 最基本的Sql查询语法
  • 2 更进一步
  • 2.1 IN 操作
  • 2.2 关系运算符:AND 和 OR
  • 2.3 排序:ORDER BY
  • 3 高级一点的话题
  • 3.1 聚合函数:COUNT,SUM, AVG
  • 3.2 对查询结果去重:DISTINCT 语法
  • 3.3 将查询数据分组:GROUP BY 语法
  • 3.4 聚合函数的好搭档:HAVING 语法
  • 4 有点超纲的话题
  • 4.1 字段类型
  • 4.2 索引
  • 4.3 JOIN语法家族
  • 4.4 嵌套的SELECT语法
  • 5 闯关答题:快速复习

Preface

为什么以《写给运营同学和初学者的Sql入门教程》为题?

这原本是给一位关系要好的运营同学定制的Sql教程。在饿了么,总部运营的同学在排查、跟踪线上问题和做运营决策的时候,除了通过运营管理系统查询信息和依赖数据分析师给出的分析数据,常常也需要直接从数据库管理台通过写Sql的方式获取更细致、更实时的业务数据,并基于这些数据进行一些及时的分析,从而更快的给出运营方案。在这样的背景下,Sql已经越来越成为我们运营同学的一项必备技能。网上有很多Sql教程(e.g. w3school),我也翻阅过一些运营同学桌上的Sql纸质书,这些教程都很好,但普遍侧重介绍语法,很多很多的语法,配以简短的demo。作为老司机的reference book很赞,可是对于刚入门甚至还没有入门的学习者,就未免困难了一点。再回顾运营同学的使用场景,大多数情况下是根据一些已有条件做一些简单的查询,偶尔会有一些相对复杂的查询,比如对查询结果做一些聚合、分组、排序,或是同时查询两三张数据表,除此以外,建表、建索引、修改表字段、修改字段值等等这些操作,在运营同学的日常工作中基本是不会遇到的。

基于以上种种原因,写了这篇教程,初衷是能够帮助这位好朋友以更高的ROI入门Sql。下面是写这篇教程时的一些考量:

  1. 从数据库、数据表的基础概念以及最简单的Sql语法开始,做一些必要的铺垫,后续的每一章再逐步增加难度;
  2. 只介绍查询语法(更准确的说,是最常用的查询语法),将它们的用法和套路解释清楚,不涉及日常工作暂时还用不到的建表、修改表结构等等操作,避免铺天盖地的语法一个接一个反而干扰了学习的节奏;
  3. 通过穿插一些小测验、小温习,及时检验对知识点的理解和复习已经学习过的知识点;
  4. 结合一些业务场景的demo帮助理解;
  5. 结尾提供一章快速复习,既是复习,也可以自测还有哪些知识点没有掌握到位;

建议所有阅读教程的同学,都尝试搭建一套自己的数据库服务(建议安装MySQL),对教程中的demo多做一些练习,不论是示例、小测验还是小温习里面的Sql语句,都不妨亲自执行一下,这也是一种很好的帮助你熟悉语法的方式。当然搭建自己的数据库会是一个不小的挑战,写作这篇教程的时候,我在自己的VPS上安装了MySQL(MariaDB)并提供了一个连接脚本(隐藏了连接MySQL的过程)给朋友使用,但是这种方式并不适合推广到所有人。具体的安装和使用方式,不在本教程的叙述范围内,所以...运营妹子们可以求助下熟悉的研发同学,汉子们嘛..

  1. 数据创建脚本-通过该脚本导入demo数据到MySQL中
  2. 链接VPS上MySQL的脚本-基本原理是建立了一个ssh tunnel,在tunnel里和远端的MySQL通信,但实际上本地建MySQL服务更方便,仅供参考

可以从这里sql_tutorial下载通过pandoc+latex导出的pdf,获得更好的阅读体验。

我也会在这次开放日活动中分享Gunicorn有关的话题,欢迎大家报名参加。放出一张分享内容的Outline:

by 多肉

Introduction

其实Sql并没有那么难。Sql是帮助你和关系型数据库交互的一套语法,主要支持的操作有4类:

  1. DQL:其实就是数据查询操作,通过特定的语法查询到数据库里你想要的数据结果,并且展示出来;
  2. DML:数据修改操作,包括更新某个数据字段、向某张表里面插入一条新的数据等等;
  3. DDL:数据定义操作,比如创建一张新的表或是创建一个新的索引(索引是啥?我们后面专门聊一聊它);
  4. DCL:数据授权操作,比如授权特定的人可以查询某张特定的表;

听起来挺吓人的对吧,但实际上DML、DDL、DCL这3类操作在日常的运营工作中几乎都不会用到,经常会使用到的呐其实是第一种,也就是数据查询操作(DQL)。Sql基本的查询语法也比较简单,那么难在哪里呢?我猜想难在学习了基本语法之后,不知道怎么应用到实际的Case上。在接下来的内容里,我将以一些十分接近现实的众包运营Case为例,逐一解释最基本的Sql查询语法并且分析如何将它应用到具体的场景上。

1 预备知识

好的吧,吹了一波牛还是逃不过需要介绍一些最基础的东西,但是我保证这是整篇教程中最枯燥的部分,后面就会有趣很多。

1.1 数据库和数据表

为了更简单的理解这两个概念以及他们之间的关系,可以这么类比:

  1. 数据表:就是一张表格,想象一张你经常在使用的Excel表,有行有列,每一行就是一条数据,每一列对应了这条数据的某一个具体的字段,当然这张表还有一个表名。数据表也是如此,只是字段名、表名不像Excel表格那样好理解,比如Excel表格里面某一列的字段名可能叫 骑手id,而对应到数据表里面可能就叫做 rider_id,Excel的表名可能叫骑手基本信息表,而对应到数据表的表名则可能叫 tb_rider;
  2. 数据库:就是集中管理一批相关的表格的地方。你可以把它理解成是一个文件夹。平时你可能会创建一个名叫 众包业务的文件夹,然后将众包运营相关的Excel表格都放在这个文件夹里面。数据库也是如此,比如我们会有一个库名叫做 crowd的数据库,里面可能存放了 tb_rider、 tb_order等和骑手、运单相关的多张数据表;

所以,“关系型数据库”的概念很吓唬人,但其实道理很简单,就是列和列之间有一定的联系,整合在一起就是一条有意义的数据,将这些数据归纳起来就构成了一张表,而将一批有关联的表一同管理起来就得到了一个数据库。

1.2 最基本的Sql查询语法

最基本的Sql查询语法其实就一个:

SELECT 列名(或者*,表示所有列) FROM 表名 WHERE 筛选条件;

让我们按照 FROM、 WHERE、 SELECT的顺序理解一下这个语法:

  1. FROM表名:顾名思义,就是从表名指定的这张表格中;
  2. WHERE筛选条件:意思是“当满足筛选条件”的时候;
  3. SELECT列名:意思是选择出这些记录,并且展示指定的列名;

串联起来便是,从FROM后面指定的数据表中,筛选出满足WHERE后面指定条件的数据,并且展示SELECT后指定的这几列字段。是不是很简单呐?不过好像抽象了一点。所以我们来看几个具体的超简单的例子。假设我们有一张学生数学期末考试成绩表,数据表长下面这样,表名叫作 tb_stu_math_score。

id(自增主键)name(学生姓名)number(学号)grade(年级)class(班级)score(得分)1柯南010201121002小哀010202121003光彦01020312984步美01020412955元太0102051259

让我们试着理解一下下面几个查询语句:

[1] SELECT name FROM tb_stu_math_score WHERE score>=95;

从 tb_stu_math_score表中挑选出得分大于95分的学生姓名,得到的结果显而易见:

| name | | ---- | | 柯南 | | 小哀 | | 光彦 | | 步美 |

[2] SELECT name,number FROM tb_stu_math_score WHERE score<60;

从 tb_stu_math_score表中挑选出得分小于60分的学生姓名,得到的结果是:

namenumber元太010205

[3] SELECT*FROM tb_stu_math_score WHERE score=100;

从 tb_stu_math_score表中挑选出得分为100分学生的所有信息(注意SELECT后面的*符号,表示所有字段),得到的结果是:


idnamenumbergradeclassscore1柯南010201121002小哀01020212100

小测验

看看下面这些Sql查询语句你是不是知道是什么含义并且知道查询结果是什么了呢?

1. SELECT name, grade, class, score FROM tb_stu_math_score WHERE number = "010201";
2. SELECT * FROM tb_stu_math_score WHERE name = "小哀";
3. SELECT id, score FROM tb_stu_math_score WHERE number = "010202";

2 更进一步

刚刚我们学习了Sql查询的最最最最基础的语法,但是相信我,所有的Sql查询几乎都长这个样子,所以理解了这个最基础的语法结构,后面学习起来就轻松多了。接下来让我通过一些例子,扩展这个基础语法,教你一些更加高级的Sql查询操作。不过首先,我们还是要看一下接下来我们的范例数据表长啥样。

假设我们有一张骑手数据表,表名叫作 tb_rider,还有一张运单数据表,表名叫作 tb_order,这两张表分别长下面这个样子。

[1] 骑手数据表: tb_rider

字段含义:

  1. id:自增主键。又是一个听起来很吓人的名字,但实际含义很简单。“自增”的意思是,每次在这张数据表中创建一条新记录的时候,数据库都会在上一个id值的基础上自动加上一个固定的步长(默认就是+1)作为新记录的id值。而所谓“主键”,就是能够在一张数据表中唯一标识一条记录的字段,因为每条记录的id都不一样,所以它是主键。这个字段可以是为了单纯的标识数据的唯一性,也可以具有一些业务含义,比如这里的id就同时也是骑手的账号id;
  2. name: 骑手姓名;
  3. real_name_certify_state: 实名认证状态:1-认证中,2-认证通过,3-认证失败;
  4. level:骑手等级,3-金牌,2-银牌,1-铜牌,0-普通;
  5. level_city:等级城市;
  6. is_deleted:这条数据是否有效,在大多数产线相关的数据表中,都会有这样的标记字段。0-未删除(表示有效), 1-已删除(表示无效);
  7. created_at:这条数据的创建时间,也是所有产线数据表中都必须有的字段;
  8. updated_at:这条数据最新一次被更新的时间,同样是所有产线数据表中都必须有的字段;

[2] 运单数据表: tb_order

字段含义:

  1. id:自增主键。呐,这里的id就是单纯的主键作用,没有其他的业务含义;
  2. order_id:运单号,业务层面上运单的唯一标识;
  3. order_state:运单当前的状态。10-待抢单,20-待到店,80-待取餐,40-已送达;
  4. rider_id:抢单的骑手id,还未被抢的运单这个字段是默认值0;
  5. rider_name:抢单的骑手姓名,还未被抢的运单这个字段是默认值空字符;
  6. grabbed_time:抢单时间,还未被抢的运单这个字段是默认的"1970-01-01 00:00:00"(这是一个特殊的时间,有兴趣的话可以搜索关键词:时间戳);
  7. created_at:这条数据的创建时间,也是所有产线数据表中都必须有的字段;
  8. updated_at:这条数据最新一次被更新的时间,同样是所有产线数据表中都必须有的字段;

小温习

试着理解看看下面这几条Sql的含义以及返回的数据结果吧?

1. SELECT name, real_name_certify_state FROM tb_rider WHERE level = 3;
2. SELECT * FROM tb_order WHERE rider_id = 1;
3. SELECT rider_id, rider_name, order_id, grabbed_time FROM tb_order
 WHERE order_state = 40;

2.1 IN 操作

场景: 线下反馈了一批骑手说自己理应是上海的金牌,但是牌级是普通或者展示的是金牌却没有享受到上海的金牌活动,你已经知道了这几个分别是id=(2, 4, 7)的骑手,想排查一下他们的等级更新情况。

这时你可以选择像这样一条一条的查询,像之前我们介绍的那样:

1. SELECT name, real_name_certify_state, level, level_city FROM tb_rider WHERE id=2;
2. SELECT name, real_name_certify_state, level, level_city FROM tb_rider WHERE id=4;
3. SELECT name, real_name_certify_state, level, level_city FROM tb_rider WHERE id=7;

这样当然可以达到目的,但是只有两三个骑手的时候还勉强可以操作,如果有几十个骑手这样查起来就太费劲了。这时候我们可以使用 IN这个语法。

SELECT name, real_name_certify_state, level, level_city FROM tb_rider WHERE id IN(2, 4, 7);

很简单的对吧?但我们还是来简单理解一下, WHERE id IN(2,4,7)的意思就是筛选id字段的值在2,4,7这几个值当中的记录,执行这条Sql语句你就会得到下面这样的结果。

namereal_name_certify_statelevellevel_cityBanner239Thor101Coulson239

于是你会发现,Thor这个骑手因为他没有通过实名认证所以肯定评不上金牌,Banner和Coulson两位骑手虽然都是金牌骑手,但是等级城市却是福州,所以享受不到上海金牌的活动。

那如果不知道骑手id,只知道骑手的名字怎么办?也可以使用 IN查询,只是这时候筛选的条件变成了 name,取值范围也变成了"Banner", "Thor", "Coulson"。就像这样。

SELECT name, real_name_certify_state, level, level_city FROM tb_rider
WHERE name IN("Banner", "Thor", "Coulson");

于是你顺利的得到了以下的结果。

namereal_name_certify_statelevellevel_cityBanner239Thor101Coulson239Coulson102

Oops! 居然有两个Coulson!

这就是在实际应用中要特别注意的地方了:

2.2 关系运算符:AND 和 OR

最常用的关系运算符有两个 AND和 OR,用来连接多个筛选条件。顾名思义, AND就是“并且”的意思,也就是同时满足 AND前后两个筛选条件; OR就是“或者”的意思,也就是满足 OR前后任何一个筛选条件。有点抽象了对不对,我们看一个具体的例子。

场景: 假设你想要看看2017-02-01(包括2017-02-01当天)到2017-06-01(不包括2017-06-01当天)期间注册的骑手所有信息。

注册时间对应到数据上就是骑手信息的创建时间( created_at),换句话说,就是查询 tb_rider``表中创建时间处于2017-02-01到2017-06-01之间的数据。那这样的Sql应该怎么写呢,这时我们就可以用到AND```。

SELECT * FROM tb_rider WHERE created_at >= "2017-02-01 00:00:00"
AND created_at < "2017-06-01 00:00:00";

让我们再来推广一下。假设现在的场景变成:想看一看2017-02-01(包括当天)之前,或者2017-06-01(包括当天)之后注册的骑手所有信息。我们应该怎么写这个Sql呢?既然是的关系,我们就应该使用 OR了。

SELECT * FROM tb_rider WHERE created_at <= "2017-02-01 00:00:00"
OR created_at >= "2017-06-01 00:00:00";

当然啦, AND和 OR这样的关系运算符,不仅仅能够连接前后两个筛选条件,也可以通过使用若干个 AND和 OR连接多个不同的筛选条件。比如:想要看看2017-02-01(包括2017-02-01当天)到2017-06-01(不包括2017-06-01当天)期间注册的且当前是金牌等级的骑手所有信息,那么我们可以这么写。

SELECT * FROM tb_rider
WHERE created_at >= "2017-02-01 00:00:00"
AND created_at < "2017-06-01 00:00:00"
AND level = 3;

2.3 排序:ORDER BY

让我们先小小的复习一下上面学到的知识点,有一个这样的场景:

如何写这个Sql呢?先思考3s...1...2...3,看看是否和你想的一样。

SELECT order_id, created_at FROM tb_order
WHERE rider_id = 1
AND grabbed_time >= "2017-12-30 00:00:00"
AND grabbed_time < "2017-12-31 00:00:00"
AND order_state = 40;

如果你没有写对,没关系,让我们来分析一下:

  1. Stark这位骑手的骑手id是1,所以我们的第一个筛选条件为 rider_id=1;
  2. 因为我们要看2017-12-30当天抢单的运单id,确定了我们的第二个筛选条件是抢单时间,对应的是 grabbed_time这个字段,而2017-12-30当天,实际上指的就是2017-12-30 00:00:00(包括)到2017-12-31 00:00:00(不包括)这段时间;
  3. 最后是已完成这个条件, order_state字段标识了运单状态,因此我们的筛选条件是 order_state=40;

执行这个语句,我们得到了下面这样的结果。

order_idcreated_at3000002017123000012017-12-30 12:34:173000002017123000022017-12-30 12:34:183000002017123000052017-12-30 16:01:03

有点美中不足,我想按照运单的创建时间倒序排序把最近创建的运单排在最前面,这时候就可以使用 ORDER BY语法了。

SELECT order_id, created_at FROM tb_order
WHERE rider_id = 1
AND grabbed_time >= "2017-12-30 00:00:00"
AND grabbed_time < "2017-12-31 00:00:00"
AND order_state = 40
ORDER BY created_at DESC;

让我们再来理解一下, DESC是“递减"的意思,与之对应的是 ASC递增。 ORDER BY created_at DESC的含义是,按照(BY) created_at字段值递减(DESC)的顺序对查询结果排序(ORDER)。于是我们得到如下的结果。

order_idcreated_at3000002017123000052017-12-30 16:01:033000002017123000022017-12-30 12:34:183000002017123000012017-12-30 12:34:17

我们再来看一个更加复杂的场景:假设想要查询2017-12-30和2017-12-31两天所有运单的所有信息,并先按照骑手id递增,再按运单状态递减的顺序排序展示。还是先思考一会儿。

这时的Sql类似长这样。

SELECT * FROM tb_order
WHERE created_at >= "2017-12-30 00:00:00"
AND created_at < "2018-01-01 00:00:00"
ORDER BY rider_id ASC, order_state DESC;

如果前面的每个知识点都理解了,这里应该就只对“先按照骑手id递增,再按运单状态递减的顺序排序展示”有所疑惑。实际上我们不仅可以对一个字段排序,还可以把多个字段作为排序的依据,而且不同字段上的排序规则(递增/递减)可以不同。但排序是有优先级的,比如这里,只有当 rider_id字段的值都相同无法区分顺序时,才会对相同 rider_id的这几条数据再按照 order_state字段的值进行排序。举例来说, rider_id=2且 order_state=80的数据,也依然不可能排在 rider_id=1且 order_state=40的数据前面。

执行这条Sql语句,将得到的结果如下。

这个部分相对有一点难,可以多对比着例子理解一下。

3 高级一点的话题

进入到这个部分,说明之前的内容你基本都已经掌握了,在日常运营的操作中有30%左右的场景都可以使用前面讲述的这些知识点解决(当然会有个熟能生巧的过程)。这个部分,我将继续介绍几个更加高级、当然也更加有难度的Sql技能,当你结束这一部分的学习并且熟练掌握这些技能的时候,你会发现绝大部分需要通过查数据来确认的场景你都可以胜任。因为这个章节的内容本身难度又大了些,如果再对着一张复杂的表就更加难以关注重点,因此我们精简一下表结构,只保留一些必要的字段。新的 tb_order表如下。

新增的列:
merchant_customer_distance:配送距离(商家到用户的直线距离),单位是千米(km)。

3.1 聚合函数:COUNT,SUM, AVG

千万别被聚合函数这个名字吓唬到,可以简单的理解为对数据进行一些加工处理,让我们先来分别看一下这几个聚合函数的基本定义。

  1. COUNT:对查询结果集合中特定的列进行计数;
  2. SUM:对查询结果的某个字段进行求和;
  3. AVG:就是average的意思,对查询结果的某个字段计算平均值;

让我们分别来看几个具体的例子。

[1] 场景:查询2017-12-30这一天,骑手Stark的所有完成单(状态为40)总量

你可以这样来写这个Sql。

SELECT COUNT(id) FROM tb_order WHERE rider_id = 1
AND order_state = 40 AND created_at >= "2017-12-30 00:00:00"
AND created_at < "2017-12-31 00:00:00";

到这里你应该已经能够很好的理解 WHERE...AND...AND...这部分的含义,我们就不再过多的讨论这个部分(对自己要有信心!试着理解先自己理解一下)。

让我们重点来看一下 COUNT(id)这部分的含义。其实很简单,就是对 id这一列进行计数。连起来看这段Sql,意思就是:从 tb_order这张表中( FROM tb_order)筛选( WHERE)骑手id为1( rider_id=1)且运单状态为已完成( order_state=40)且创建时间大于等于2017年12月30日( created_at>="2017-12-30 00:00:00)且创建时间小于2017年12月31日( created_at<"2017-12-31 00:00:00)的数据,并且按照 id这列对返回的结果集合进行计数。

我们看到 tb_order这张表中,2017-12-30当天由骑手Stark配送且状态是已完成的运单分别是300000201712300001、300000201712300002、300000201712300005这几个运单号的运单,对应的自增id分别是id=[1, 2, 5],所以对 id这一列进行计数得到的结果是3。所以我们得到的查询结果如下表。

| COUNT(id) | | --------- | | 3 |

有时候你仅仅是想查一下满足某个条件的记录的总行数,而并非想对某个特定的列进行计数,这时就可以使用 COUNT(*)语法。比如上面的这个Sql也可以写成下面这个样子。

SELECT COUNT(*) FROM tb_order WHERE rider_id = 1
AND order_state = 40 AND created_at >= "2017-12-30 00:00:00"
AND created_at < "2017-12-31 00:00:00";

因为返回的结果有三行,所以我们会得到下表的结果。

| COUNT(*) | | -------- | | 3 |

看起来 COUNT(列)和 COUNT(*)是完全等价的?有些特定的场景下的确如此,这里需要补充一下COUNT的两个小脾气。

  1. COUNT不会自动去重;
  2. COUNT在某一条查询结果中,用来计数的那一列的值为“空"时,这条记录不进行计数;

有一点晕是吗?不着急,我们来看两个例子。假设有两张表,很简单的表,长下面这样。

示例表1: tb_sample_1

idname1Stark2Stark3Coulson4Natasha5Stark

示例表2: tb_sample_2

idname1Stark2Stark3\4\5Natasha6Coulson

我们下猜一猜下面几条Sql的执行结果分别是什么?

1. SELECT COUNT(id) FROM tb_sample_1;
2. SELECT COUNT(*) FROM tb_sample_1;
3. SELECT COUNT(name) FROM tb_sample_1;
4. SELECT COUNT(name) FROM tb_sample_2;

让我们逐一分析一下。

1. SELECT COUNT(id) FROM tb_sample_1;

这条Sql没有太多可以分析的,因为 tb_sample_1表中 id字段的取值范围是id=[1, 2, 3, 4, 5],共5个,所以我们得到的结果如下。

| COUNT(id) | | --------- | | 5 |

2. SELECT COUNT(*) FROM tb_sample_1;

这条Sql也没有太多需要分析的,因为 COUNT(*)的含义是计算查询结果的总行数, tb_sample_1共5行数据,所以我们得到的结果如下。

| COUNT(*) | | -------- | | 5 |

3. SELECT COUNT(name) FROM tb_sample_1;

这条Sql里面我们对 name这一列进行计数, tb_sample_1表中包含3个Stark,1个Coulson和1个Natasha,因为COUNT不进行自动去重,因此结果是 5=3(Stark)+1(Coulson)+1(Natasha),如下表。

| COUNT(name) | | ----------- | | 5 |

4. SELECT COUNT(name) FROM tb_sample_2;

这条Sql语句我们还是对 name这一列进行计数, tb_sample_2表中包含2个Stark,1个Coulson,1个Natasha以及2个 ,由于COUNT不去重因此2个Stark都会被计数,但COUNT不会对值为“空”的结果进行计数,因此两个 都会被忽略。所以最终的结果为 4=2(Stark)+1(Coulson)+1(Natasha),如下表。

| COUNT(name) | | ----------- | | 4 |

[2] 场景:查询Stark这名骑手的累计配送里程

让我们先定义一下累计配送里程:骑手所有配送完成单的配送距离(商家到用户的直线距离)之和。

这里的关键词是求和,所以我们要用到 SUM这个聚合函数。对字段求和的意思是把返回的结果集合中该字段的值累加起来。让我们看下这个场景的Sql怎么写。

SELECT SUM(merchant_customer_distance) FROM tb_order
WHERE rider_id = 1 AND order_state = 40;

让我们来分析一下这条语句, FROM tb_order WHERE rider_id=1AND order_state=40已经比较好理解了,就是从 tb_order表中筛选出骑手id为1且配送状态为40的记录。而这里的 SUM(
merchant_customer_distance)的含义,就是对前面的条件筛选出的数据结果中的
merchant_customer_distance列的值进行求和。根据骑手id和配送状态筛选出的记录分别为id=(1, 2, 5),对应的
merchant_customer_distance的值分别为merchantcustomerdistance=(2.5, 1.8, 1.2),求和结果为5.5=2.5+1.8+1.2,如下表。

| SUM(merchant_customer_distance) | | --------------------------------- | | 5.5 |

[3] 场景:查询Stark这名骑手的平均配送里程

同样的,让我们先来定义一下平均配送里程:骑手所有完成单的配送距离(商家到用户的直线距离)之和除以总的完成单量。

基于 SUM的经验和前面的“预告”,不难想到这次我们会用到 AVG这个聚合函数。对字段求平均值的意思是,把结果集合中该字段的值累加起来再除以结果总行数。 AVG帮我们自动完成了“做除法”的动作,所以Sql的长相和上一个场景的 SUM是如出一辙的。

SELECT AVG(merchant_customer_distance) FROM tb_order
WHERE rider_id = 1 AND order_state = 40;

根据骑手id和配送状态筛选出的记录分别为id=(1, 2, 5),对应的
merchant_customer_distance的值分别为merchantcustomerdistance=(2.5, 1.8, 1.2),求平均值的结果为1.83=(2.5+1.8+1.2) / 3,如下表。

| AVG(merchant_customer_distance) | | --------------------------------- | | 1.83 |

写在3.1节的最后:

对着这几个场景学习下来,不知道你感觉怎么样吖?是否觉得这几个聚合函数本身还蛮简单的,或者也有可能会觉得一下子灌输了很多知识点有点费劲呢?其实聚合函数有它复杂的一面,我们上面看的这些Case都是比较简单的使用方式。但是千万不要担心,一方面是因为运营工作中遇到的绝大多数场景都不会比这些示例Case更复杂,另一方面是不鼓励过于复杂的使用这些聚合函数,因为查询的逻辑越是复杂就越是难以“预测”查询的结果,Sql并不是一个适合表达“逻辑”的语言,如果对数据的再加工逻辑很多,就应该考虑像分析师提需求或者学习更加利于表达逻辑的其他编程语言。

其次要说的就是多给自己些信心,同时也要多一点耐心。Sql虽然不同于Python、Java这样的通用编成语言,除了语法还杂糅着一套体系化的编程概念、设计哲学,但是初次上手的时候还是会感觉到有些吃力的。但是只要多去理解几遍示例、多自己写一写,特别是在之后遇到实际工作中真实场景的时候自己思考如何转化为Sql、多实践、多回顾分析,很快就会在潜移默化中掌握它,要相信熟能生巧。

接下来的3.2、3.3节,我会继续介绍两个实用的Sql语法,以及如何将它们和聚合函数结合使用,会更有难度一些。

3.2 对查询结果去重:DISTINCT 语法

DISTINCT语法顾名思义就是对某一列的值进行去重,让我们首先来回顾一下3.1节中COUNT的其中一个例子。

这个例子使用的是 tb_sample_1这张表,这张表很简单,让我再把它贴出来。

idname1Stark2Stark3Coulson4Natasha5Stark

对应的,我们想要回顾的这条Sql语句也很简单。

SELECT COUNT(name) FROM tb_sample_1;

前面我们已经分析过这条Sql:对 name这列进行计数,有3个Stark,1个Coulson,1个Natasha,所以得到最终的结果如下表。

| COUNT(name) | | ----------- | | 5 |

可是有的时候,我们不想对相同的名字进行重复计数,当有多个相同的名字时只计数一次。这时候就可以使用到DISTINCT语法。

SELECT COUNT(DISTINCT name) FROM tb_sample_1;

对比上一条Sql只是增加了一个DISTINCT关键字,其实理解起来呢也不用把它想的太复杂啦: COUNT(DISTINCT name)就是对去重后的 name进行计数。 tb_sample_1中有3个Stark,但是3个Stark是重复的,使用DISTINCT语法后只会被计算一次,另外还有1个Coulson和一个Natasha,所以得到的结果如下表。

| COUNT(DISTINCT name) | | -------------------- | | 3 |

DISTINCT语法可以单独使用,这时就是它本身的意思,对某列的值进行去重。但是相比之下,更常见的是像上面的例子一样和COUNT这个聚合函数一起使用,这样就可以对去重后的结果进行计数。

3.3 将查询数据分组:GROUP BY 语法

前面我们基于 tb_order这张表讲解了很多Sql的语法知识,让我们再来回忆一下这张表的容颜。

温故而知新!先来出几道题目复习一下前面所学的Sql知识。

复习题1: 试着写出以下几个场景对应的Sql语句

  1. 查询2017-12-30当天创建的运单,状态为已完成且配送距离大于等于2公里的总单量;
  2. 查询2017-12-30当天创建且状态为已完成的所有运单的平均配送距离;
  3. 查询2017-12-30当天完成过配送任务(至少配送完成1单)的骑手总人数;

复习题2: 试着理解以下几条Sql的含义并且写出查询的结果

1. SELECT COUNT(order_id) FROM tb_order WHERE order_state = 40
 AND merchant_customer_distance >= 2.0 AND created_at >= "2017-12-30 00:00:00"
 AND created_at < "2017-12-31 00:00:00";
2. SELECT AVG(merchant_customer_distance) FROM tb_order WHERE order_state = 40
 AND created_at >= "2017-12-30 00:00:00" AND created_at < "2017-12-31 00:00:00";
3. SELECT COUNT(DISTINCT rider_id) FROM tb_order WHERE order_state = 40
 AND created_at >= "2017-12-30 00:00:00" AND created_at < "2017-12-31 00:00:00";

聪明的你是否发现复习题2就是复习题1的答案呢?如果还没有发现,没关系,再回过头来多分析几遍,Practice Makes Perfect 绝对是真理。不过复习这几个例子可不仅仅是为了复习哦,让我们在1、2两个场景的基础下扩展一下,讲解新的知识点。思考下面这两个场景。

  1. 查询2017-12-30当天每个参与跑单的骑手各自的完成单总量;
  2. 查询2017-12-30当天每个参与跑单骑手的完成单平均配送距离;

首先分析一下这里的场景1。“2017-12-30当天”这个条件不难转化为 created_at>='2017-12-30 00:00:00'AND created_at<'2017-12-31 00:00:00',“完成单”不难转化为 order_state=40,由于要计算运单的“总量”我们也不难想到可以对 order_id进行COUNT操作。那么如何分组到每个骑手身上呢?这时候就要用到GROUP BY了。

SELECT COUNT(order_id) FROM tb_order WHERE order_state = 40
AND created_at >= "2017-12-30 00:00:00" AND created_at < "2017-12-31 00:00:00"
GROUP BY rider_id;

注意这里执行顺序是先按照WHERE条件进行筛选,然后根据骑手id进行分组(GROUP BY),最后再对每个分组按照运单号进行计数。因此我们可以得到下表的结果。

| COUNT(order_id) | | ------------------- | | 3 | | 1 | | 1 | | 1 |

好像有哪里不对?结果中看不到对应的骑手吖!不着急,我们稍微修改下刚才的Sql,将骑手id、骑手姓名这2列展示出来就可以了。

SELECT rider_id, rider_name, COUNT(order_id)
FROM tb_order WHERE order_state = 40
AND created_at >= "2017-12-30 00:00:00"
AND created_at < "2017-12-31 00:00:00"
GROUP BY rider_id;

我们得到如下表的结果。

rider_idrider_nameCOUNT(order_id)1Stark32Banner15Natasha13Rogers1

这样是不是就清晰多了。

再来分析场景2。有了前面的例子,“2017-12-30当天”、“完成单”这两个条件应该是已经得心应手、信手拈来了,“平均配送距离”问题也不大,可以转化为 AVG(
merchant_customer_distance)。那么如何分组到每个骑手身上呢?还是通过 GROUP BY语法。我们的Sql长成下面这个样子。

SELECT rider_id, rider_name, AVG(merchant_customer_distance)
FROM tb_order WHERE order_state = 40
AND created_at >= "2017-12-30 00:00:00"
AND created_at < "2017-12-31 00:00:00"
GROUP BY rider_id;

得到如下表的结果。

rider_idrider_nameAVG(merchant_customer_distance)1Stark1.832Banner1.85Natasha2.73Rogers0.5

还是需要特别提一下这里的执行顺序,首先执行的是 WHERE条件筛选,然后对筛选出的数据结果根据骑手id进行分组,最后再对每个分组中的数据进行
merchant_customer_distance列的求平均值。

3.4 聚合函数的好搭档:HAVING 语法

HAVING语法的含义类似于WHERE,当我们使用HAVING的时候一般遵循 HAVING筛选条件的语法结构。你可能会问啦,既然和WHERE语法含义差不多、使用方式又很类似,那干嘛还要凭空多个HAVING语法出来呢?原因就在于聚合函数。WHERE语法是不能和聚合函数一起使用的,但有些时候我们却需要依赖聚合函数的计算结果作为筛选条件。让我们看一下3.3节中场景2这个例子。

场景2:查询2017-12-30当天每个参与跑单骑手的完成单平均配送距离。

通过前面我们的分析,得到这样的Sql。

SELECT rider_id, rider_name, AVG(merchant_customer_distance)
FROM tb_order WHERE order_state = 40
AND created_at >= "2017-12-30 00:00:00"
AND created_at < "2017-12-31 00:00:00"
GROUP BY rider_id;

我们在场景2的基础上再扩展一下。

扩展的场景2:查询2017-12-30当天每个参与跑单骑手的完成单平均配送距离,并筛选出其中平均配送距离超过1.5km的数据。

我们得到这样的Sql结果。

SELECT rider_id, rider_name, AVG(merchant_customer_distance)
FROM tb_order WHERE order_state = 40
AND created_at >= "2017-12-30 00:00:00"
AND created_at < "2017-12-31 00:00:00"
GROUP BY rider_id
HAVING AVG(merchant_customer_distance) > 1.5;

比较一下不难发现,变化仅仅是末尾多了 HAVING AVG(
merchant_customer_distance)>1.5这条子句。让我们分析看看。 SELECT...FROM...WHERE...和之前的用法并没有变化, GROUP BY rider_id将SELECT的结果根据 rider_id进行分组,分组完成后 HAVING AVG(
merchant_customer_distance)>1.5语句对每一组的
merchant_customer_distance字段值求取平均数,并且将平均数大于1.5的结果筛选出来,作为返回结果。

执行这条Sql我们得到结果。

rider_idrider_nameAVG(merchant_customer_distance)1Stark1.832Banner1.85Natasha2.7

Rogers这位骑手(骑手id=3)因为平均配送距离为0.5,不满足HAVING语句指定的“平均配送距离大于1.5km”的筛选条件,所以没有在我们的查询结果中。

4 有点超纲的话题

4.1 字段类型

类型这个词此刻你听起来可能还是很陌生的,但其实在计算机科学领域,类型是一个非常基础而且广泛存在的概念,几乎每一种编程语言都有自己的类型系统。

[1] 为什么要定义类型的概念?

关于为什么要有类型这个概念,我呐有一个“不成熟”的理解:编程语言作为人和机器交互的一种工具,人类对数据有人类逻辑上的理解,当我们看到2903的时候我们会认为这是个整数,当我们看到1031.2903的时候我们会认为这是个小数。而机器在处理数据或者存取数据的时候,是无差别的按照比特位进行二进制运算或者读写的。人类很难做到直接用二进制输入计算机,当然也不能接受计算机直接以二进制的形式输出结果。设想一下,如果某天咱们想用一下电脑上的计算器,计算个 1+1=2,但是我们没有类型,我们需要理解机器是如何处理二进制的,那么就可能需要输入
00000000000000000000000000000001+
00000000000000000000000000000001,而得到的结果也是二进制
00000000000000000000000000000010,这得多累人呐。有了类型就轻松多了,通过定义数据的类型,根据类型的约定,计算机就知道如何将这个1转化为二进制(包括:应该转化为16位、32位还是64位的二进制,对这段二进制数据进行操作的时候,应该把它看作整数还是浮点数等等),而返回结果的时候也就知道如何将二进制的
00000000000000000000000000000010转化为我们能够理解的整数2

编程语言的类型其实就是人与机器约定好的,去理解和操作数据的一套规则

总而言之,在机器的眼里,无论是对数据进行何种操作,它看到的都是一串一串由0和1构成的东西,称呼这种东西有专门的术语,叫作“字节流”或者“二进制流“

让我们再一起看一个例子。假设要处理这样的一段二进制流:
00000000100111011000001111010111,这段二进制流可以表示很多东西,要明确它的含义,就需要明确它的类型,比如下面这两种不同的类型,这段流表示的内容就完全不同。

  1. 如果我们把这段二进制流看作是32位整型,那么它代表的是10322903这个整数;
  2. 如果我们把这段二进制流看作是2个16位整型(前16位 0000000010011101表示一个整型,后16位 1000001111010111表示一个整型),那么它分别代表的是157和33751这两个整数;

我知道你此刻对为何转换为32位整型是10322903?为何看作2个16位整型转换后是157和33751?还有着很多疑惑。但是关于二进制和十进制的转换方法呢,在这里就不做展开了,如果你很感兴趣、很想知道可以再单独给你讲这个方法。讲上面的这些,最主要的还是希望你明白,定义“类型”的概念,根本上是在人机交互的过程中提供了一种机制,赋予无差别的二进制流一定的语义

还是太抽象了对不对?没关系,我们再来举个栗子。

前面我们在预备知识这一章中使用到了 tb_stu_math_score这张表,为了不让你辛苦的再翻回去,我们再贴一下这张表的内容啦。

id(自增主键)name(学生姓名)number(学号)grade(年级)class(班级)score(得分)1柯南010201121002小哀010202121003光彦01020312984步美01020412955元太0102051259

也写过类似下面这条Sql语句。

SELECT score FROM tb_stu_math_score WHERE id=1;

这条Sql语句非常非常的简单,现在我们已经知道它会返回第一行数据 score这一列的值,结果长下面这样。

| score | | ----- | | 100 |

sql数据运营(写给运营同学和初学者的 SQL 入门教程)

让我们分析一下获取这个结果的整个流程,帮助你理解一下,类型是如何发挥作用的。

  1. 这条Sql语句会被执行,根据主键id找到对应的行,进而获取到这一行、 score这列的值;
  2. 但是我们上面说到,计算机的存储是基于二进制的,所以获取到的 score值是类似于 00000000000000000000000001100100这样的二进制流;
  3. 这时候根据 tb_stu_math_score表的定义, score这一列被定义为整型,于是将二进制流转化为一个整型数,经过进制转换得到 00000000000000000000000001100100对应的整型值为100,于是我们看到的结果就是100。

实际上反过来也非常类似,当我们向这张表中写入数据时,例如写入的 score列的值为100。因为存储基于二进制,根据表的定义, score列的类型为整型,于是将值100按照整型转换为对应的二进制流
00000000000000000000000001100100,并且写入到库中。

[2] Sql的主要数据类型有哪些?

Sql中常常接触的数据类型主要包括几类。

1 整型

  1. tinyint:用来表示很小很小的整数,比如常常用它作为 is_deleted、 is_valid这些字段的字段类型,因为这两个字段表示该条记录是否有效,只存在两个值分别是0和1;
  2. smallint:比 tinyint稍微大一点点的整型,可以表示更大一点的整数,比如200、404、401这样的整数值;
  3. int:常用的整型,可以用来表示比较大的整数,比如10322(事实上 int可以表示的整数范围远远比这个大);
  4. bigint:用来表示非常大的整数,比如大多数表的自增id就会使用这个类型,可以表示类似10322903这样非常大的整数(事实上 bigint可以表示的整数范围远远比这个要大);

2 浮点型

  1. decimal:可以表示非常准确的小数,比如经纬度;

3 字符串类型

  1. char:固定长度的字符串;
  2. varchar:可变长度的字符串;

这里固定长度和可变长度指的是数据库中的存储形式,因为这部分的内容其实有些超出了这个教程的范围,我们不过多的解释这里的区别。一般在我们实际的应用中 varchar用的更多一些。它们都表示类似于 "very glad to meet u, Huohuo!"这样的一串字符,当然也可以是中文 "敲开心认识你,火火!"。

4 日期类型

  1. date:表示一个日期,只包含日期部分,不包含时间,比如当前日期"2018-01-23";
  2. datetime:表示一个日期,同时包含日期部分和时间部分,比如当前日期"2018-01-23 03:01:43";

我们在这里只是简单的介绍了几种Sql中常见的字段类型,并没有很深入的去解释它们的原理、差异以及一些其他的数据类型,咱们不着急去学习那些“高大上”的内容,先理解这些类型的含义。

[3] 怎么知道一张表中每一列的类型是什么?

第1种方式是使用 DESC表名命令,例如我们想看一下之前提到的 tb_rider表的每一列字段类型,就可以执行命令 DESC tb_rider,得到下面的结果。

注意这里的第一列表示字段名称,第二列 Type则表示对应字段的字段类型。比如 id字段,是一个 int类型。

第二种方式是使用 SHOW CREATE TABLE表名命令,例如 SHOW CREATE TABLE tb_rider,得到下面的结果。

CREATE TABLE `tb_rider` (
 `id` int(11) NOT NULL AUTO_INCREMENT,
 `name` varchar(32) NOT NULL DEFAULT '' COMMENT '姓名',
 `real_name_certify_state` int(11) NOT NULL DEFAULT '0' COMMENT '身份证认证状态',
 `is_deleted` tinyint(4) NOT NULL DEFAULT '0' COMMENT '该用户是否还存在. 0: 不存在, 1: 存在',
 `created_at` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
 `updated_at` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '更新时间',
 `level` tinyint(4) NOT NULL DEFAULT '0' COMMENT '骑手等级:0普通 1铜牌 2银牌 3金牌',
 `level_city` varchar(32) NOT NULL DEFAULT '' COMMENT '配送员等级城市',
 PRIMARY KEY (`id`),
 KEY `ix_created_at` (`created_at`),
 KEY `ix_updated_at` (`updated_at`)
) ENGINE=InnoDB AUTO_INCREMENT=9 DEFAULT CHARSET=utf8 COMMENT='配送员信息';

我们以

`name` varchar(32) NOT NULL DEFAULT '' COMMENT '姓名'

来解释一下这里的语句。

  1. name是字段名(列名);
  2. varchar表示字段类型为字符串;
  3. NOT NULL表示这个字段不能为空, 为空的意思是没有指定任何值给这个字段(注意不等价于空字符串);
  4. ``DEFAULT''表示如果没有指定这个字段的值则使用空字符串作为默认值;
  5. COMMENT'姓名'是对这个字段的备注,表示这个字段的业务含义,只用做展示;

4.2 索引

索引绝对算得上是关系型数据库中最关键同时也是最有难度的话题。即便是经验丰富的研发同学,也经常会踩到索引的坑。不过我们这里介绍索引,只是为了更好的服务于查询,我会尽可能避免牵扯进一些复杂的概念和底层原理。

[1] 什么是索引?

那么到底什么是索引呢?你可以把数据库理解为一本很厚的书(假设有10万页),书中的内容就是数据库里的数据,那么索引就是书的目录。 假设你从来没有阅读过这本书,此刻你想要阅读书的第7章第2小节。如果没有目录,你可能需要翻阅整本书找到你要阅读的内容。但是在有目录的情况下,你就只需要先查一下目录找到对应的页码,然后直接翻到那一页就能看到你想看的内容了。索引也是类似的,首先查询索引找到目标数据的位置,再从特定的位置读取出数据的内容。

如何设计索引,是设计数据库表的时候考虑的关键点之一。索引一般由表中的某一列或者某几列构成,一旦设置某一列为索引,那么之后每次在往表中写入数据的时候,都会更新这一列到索引中去。事实上,索引在技术层面是比较复杂的,涉及到磁盘I/O、B树、优化器(Optimizer)等很多技术概念,不过我们先不去深究这些。

[2] 为什么索引很重要,它有什么用?

索引之所以重要,最主要的原因是能够大大提高查询的速度。上面我们举了书的例子,当这本书的页数足够大的时候(假设有2000万页),如果没有目录,想要查阅其中的某一章节的内容,那几乎就是天方夜谭了。数据库也是如此,当表中的数据只有几行或者几十行、几百行的时候,有没有索引其实差别不大,但是当表中的数据非常非常多的时候(比如众包的运单表,2000万+ 行),如果没有索引,要找到某一条目标数据,查询的速度就会非常非常非常的慢。

[3] 如何使用索引?

要使用索引非常简单,只需要在 WHERE条件中使用到索引列作为查询条件,让我们举个例子。

还是这张 tb_order表,假设这张数据表中 order_id是索引列,那么当我们以 order_id作为查询条件时,我们就利用了索引,比如下面这条Sql。

SELECT * FROM tb_order WHERE order_id = 300000201712310007;

当然啦,类似的使用 order_id作为查询条件的Sql也都会利用到索引,看看你是否都理解下面两条Sql语句的含义。

1. SELECT * FROM tb_order
 WHERE order_id IN (300000201712310007, 300000201712310006)
 AND order_state = 40;
2. SELECT order_id, order_state FROM tb_order
 WHERE order_id >= 300000201712300001
 AND order_id <= 300000201712300006
 AND order_state = 40;

那么如果一张表里面不止一列是索引,而在查询的Sql中这些索引列都作为了 WHERE语句的查询条件,会使用哪个列作为索引还是都使用?假设 tb_order表中 order_id和 rider_id两列都是索引列,那么下面这条Sql语句会使用哪个作为索引呢?

SELECT * FROM tb_order
WHERE order_id >= 300000201712310001
AND order_id <= 300000201712310007
AND rider_id > 0;

答案是不确定的。使用哪个索引,甚至是否使用索引,从根本上来说是由优化器(Optimizer)决定的,它会分析多个索引的优劣,以及使用索引和不使用索引的优劣,然后选择最优的方式执行查询。这部分话题就太过复杂了,这里不做展开。尽管有优化器(Optimizer)的存在,但是对于我们的查询来说,能够使用明确的索引字段作为查询条件的,就应该尽可能使用索引字段。

[4] 索引的类型、如何确定表中的哪些列是索引列?

还记得字段类型一节中提到的 DESC表名和 SHOW CREATE TABLE表名语法吗?前面我们将这两个语法用在了 tb_rider表上,这一节让我们看一看 tb_order表。

首先是 DESC tb_order,我们会得到下面的结果。

之前我们关注的是 Type这一项,这里让我们关注 Key这一项。我们看到有些列对应的 Key是空的,这就表示这一列(或者叫这个字段)不是索引列(或者叫索引字段)。但 id、 order_id、 created_at和 updated_at这几列对应的 Key均是有值的,这说明这几列都是索引列。但这几列 Key的值又各不相同,这是为啥呐?这是以内索引也分为不同的类型,让我们逐个来解释一下。

  1. PRI:是primary的缩写,标记这一列为主键,主键的概念我们在一开始的时候有介绍过,就是用来唯一标识表中每一行数据的索引;
  2. UNI: 是unique的缩写,顾名思义就是唯一的意思, 被设置为 UNI KEY的列,不允许出现重复的值,如果尝试向表中插入这一列的值完全相同的两行数据,则会引发报错。我猜你肯定会觉得疑惑,那 UNI KEY和 PRI KEY有啥区别?首先是这两种类型的索引在实现上是有区别的(这一点咱们不深究,涉及到了数据库底层对索引的实现),其次 PRI KEY更多的是数据库层面的语义,仅仅是描述数据的唯一性,而 UNI KEY则更多是业务层面的语义,比如说这里的 order_id字段,因为业务上不能存在两个运单号完全相同的运单,所以需要把 order_id这一列设置为 UNI KEY;
  3. MUL:是multiple的缩写,表示这一列是被设置为一个普通索引。之所以叫做 multiple,是因为此时可能这一列单独作为索引,也可能这一列和其他标记为 MUL的列共同构成了一个索引(这种由多列共同构成的索引被叫作复合索引);

现在我们还处在Sql以及数据库知识(是的,除了Sql,我还偷偷介绍了一些数据库原理)学习的初级阶段,所以让我们知道这写差异,但是不着急去把这些搞得一清二楚,它们都是索引,只要合理使用,都可以帮助我们加快Sql查询的效率。

另一种识别表中索引列的方法就是通过 SHOW CREATE TABLE表名命令,比如 SHOW CREATE TABLE tb_order,我们得到下面的结果。

CREATE TABLE `tb_order` (
 `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '对外不提供,内部使用',
 `order_id` bigint(20) NOT NULL DEFAULT '0' COMMENT '运单的跟踪号(可以对外提供)',
 `rider_id` int(11) NOT NULL DEFAULT '0' COMMENT '配送员id',
 `rider_name` varchar(100) NOT NULL DEFAULT '' COMMENT '配送员名字',
 `order_state` tinyint(4) NOT NULL DEFAULT '0' COMMENT '配送状态',
 `is_deleted` tinyint(4) NOT NULL DEFAULT '0',
 `grabbed_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '抢单时间',
 `merchant_customer_distance` decimal(10,2) NOT NULL DEFAULT '0.00' COMMENT '商铺到顾客步行距离',
 `created_at` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP,
 `updated_at` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
 PRIMARY KEY (`id`),
 UNIQUE KEY `uk_order_id` (`order_id`),
 KEY `ix_created_at` (`created_at`),
 KEY `ix_updated_at` (`updated_at`)
) ENGINE=InnoDB AUTO_INCREMENT=14 DEFAULT CHARSET=utf8 COMMENT='配送单';

看到末尾几行的 PRIMARY KEY、 UNIQUE KEY和 KEY了吗,它们就对应于 DESC tb_order结果中的 PRI、 UNI和 MUL,分别标识主键索引、唯一索引和普通索引。每一行括号内的字段就表示对应的索引列。

4.3 JOIN语法家族

我尝试了好几种解释清楚JOIN语法的方法(JOIN语法的确有些复杂),始终不能让我自己满意,最终决定还是从一个例子开始。让我们首先看一张新的表,建表语句长下面这样。

CREATE TABLE `tb_grab_order_limit` (
 `id` BIGINT(20) NOT NULL AUTO_INCREMENT COMMENT '自增主键',
 `rider_id` BIGINT(20) NOT NULL DEFAULT 0 COMMENT '骑手id',
 `order_grab_limit` INT(11) NOT NULL DEFAULT '0' COMMENT '接单上限',
 `is_deleted` TINYINT NOT NULL DEFAULT 0 COMMENT '该记录是否被删除',
 `created_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
 `updated_at` DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '更新时间',
 PRIMARY KEY(`id`),
 KEY `ix_rider_id` (`rider_id`),
 KEY `ix_created_at` (`created_at`),
 KEY `ix_updated_at` (`updated_at`)
) ENGINE = InnoDB DEFAULT CHARSET=utf8 comment="自定义骑手接单上限表";

小温习

参考上面的建表语句尝试回答下面这几个问题。

  1. 这张表的表名是什么?
  2. order_grab_limit这个字段的含义是什么?
  3. 这张表的主键索引是什么?有几个唯一索引、几个普通索引?

没错!这就是自定义骑手接单上限表。描述了某一个骑手( rider_id)对应的他的接单上限( order_grab_limit)。表中的数据如下。

idrider_idorder_grab_limitis_deletedcreated_atupdated_at111102018-02-25 17:22:032018-02-25 17:22:0322902018-02-25 17:22:212018-02-25 17:22:2134902018-02-25 17:22:312018-02-25 17:22:3146702018-02-25 17:22:392018-02-25 17:22:39510802018-02-25 17:22:462018-02-25 17:22:46

再让我们回顾一下前面反复用到的 tb_rider表。

(终于铺垫完啦!)

[1] 从LEFT JOIN开始

以这两张表为基础,设想一个场景:假设要查询 tb_rider表中所有骑手对应的自定义接单上限。我们的Sql应该怎么写呢?

思路1:先查出 tb_rider表中所有骑手id,再根据这些骑手id作为查询条件,通过前面学习过的IN语法从 tb_grab_order_limit表中查询出所对应的自定义接单上限的记录。

SELECT id FROM tb_rider;

SELECT rider_id, order_grab_limit FROM tb_grab_order_limit
WHERE rider_id IN (1, 2, 3, 4, 5, 6, 7, 8);

思路1显然是个Bad idea。但是思路1诠释了解决这个查询问题的基本要点。

  1. 我们最终想要的数据是需要结合 tb_rider和 tb_grab_order_limit两张表共同得出的;
  2. 关联这两张数据表的条件是骑手id;
  3. 因为查询的要求是: tb_rider表中所有骑手,因此应该以 tb_rider表中的骑手id作为查询参考集合;
  4. 不是所有 tb_rider表中的骑手都配置了自定义接单上限,思路1的查询方案存在一个缺点,就是我们需要根据查询结果,在逻辑上做一个转换得知哪些骑手没有配置自定义接单上限( 不在返回结果中的骑手);

思路2:基于这几个要点我们可以使用LEFT JOIN语法,下面是对应的Sql语句。

SELECT tb_rider.id, tb_grab_order_limit.order_grab_limit
FROM tb_rider LEFT JOIN tb_grab_order_limit
ON tb_rider.id = tb_grab_order_limit.rider_id;

这里先介绍一下JOIN语法的基本结构: 表1(INNER/LEFT/RIGHT/FULL)JOIN表2ON表1.列1=表2.列2。JOIN关键字前后连接的是两张需要关联查询的数据表,ON关键字后面跟着关联的条件。一共有四种类型的JOIN,他们分别是INNER JOIN、LEFT JOIN、RIGHT JOIN和FULL JOIN。以例子中的LEFT JOIN为例, 表1LEFT JOIN表2ON表1.列1=表2.列2的含义是,遍历表1中的列1的值,如果表2中列2的值有和它相等的则展示对应的记录,如果没有表2.列2和表1.列1相等,则展示为null。

思路2的例子中, tb_rider LEFT JOIN tb_grab_order_limit ON tb_rider.id=
tb_grab_order_limit.rider_id的含义是,遍历 tb_rider表中 id这一列( tb_rider表的 id字段业务含义就是骑手id)的值,寻找 tb_grab_order_limit表中 rider_id列的值和它相等的记录,如果不存在则是null。

我们还看到SELECT语句的内容和我们之前使用的很类似,但又稍微有点不一样,都是表名.列名的书写形式。其实这主要是指明了字段所属的表,因为JOIN的两张数据表中可能存在的相同名称的列,例如 tb_rider表和 tb_grab_order_limit表都有 id字段,但含义截然不同,这样写更加明确。

最终思路2的结果如下。

idorder_grab_limit1112949677\8\5\3\

我们看到骑手id=(7, 8, 5, 3)的几个骑手没有配置自定义的接单上限,但因为是LEFT JOIN,他们仍然会展示在查询结果中,不过因为没有接单上限的记录, order_grab_limit的结果为null。

让我们再回头看一下表名.列名这个写法。如果思路2中的Sql改成下面这样,返回结果会变成什么呢?

SELECT tb_grab_order_limit.rider_id, tb_grab_order_limit.order_grab_limit
FROM tb_rider LEFT JOIN tb_grab_order_limit
ON tb_rider.id = tb_grab_order_limit.rider_id;

让我们来分析一下。我们知道LEFT JOIN的返回结果集合是以它左侧连接的数据表决定的,所以结果集仍然包含8条记录,但是骑手id=(7, 8, 5, 3)这个骑手没有对应的接单上限的配置,因此当我们展示这几个骑手的
tb_grab_order_limit.rider_id列的值的时候,类似于
tb_grab_order_limit.order_grab_limit,也是null。因此结果是下面这样。

rider_idorder_grab_limit111294967\\\\\\\\

如果你还是不太明白,然我们在SELECT的时候,加上 tb_rider.id,或许有助于理解。

SELECT tb_rider.id, tb_grab_order_limit.rider_id, tb_grab_order_limit.order_grab_limit
FROM tb_rider LEFT JOIN tb_grab_order_limit
ON tb_rider.id = tb_grab_order_limit.rider_id;

结果是。

idrider_idorder_grab_limit11112294496677\\8\\5\\3\\

[2] LEFT JOIN的姊妹篇:RIGHT JOIN

前面我们知道LEFT JOIN是以连接的左侧表作为查询的结果集的依据,RIGHT JOIN则是以连接的右侧表作为依据。让我们考虑另一个场景:假设想要查询所有设置了自定义接单上限的骑手姓名。应该如何写这个Sql呢?

先在聪明的大脑里思考几分钟。此时你需要类比LEFT JOIN,需要理解上一段内容讲述的LEFT JOIN知识点,可能需要回到上一段再看一看示例Sql语句以及对应的结果。没关系,一开始学习的时候慢慢来。

答案是这样的。

SELECT tb_grab_order_limit.rider_id, tb_rider.name
FROM tb_rider RIGHT JOIN tb_grab_order_limit
ON tb_rider.id = tb_grab_order_limit.rider_id;

对应的查询结果则是。

rider_idname1Stark2Banner4Thor6Barton10\

如果这个结果和你脑海中思考的结果不一样,不要着急,让我们再来解释一下。RIGHT JOIN是以连接的右侧表为依据,而 tb_grab_order_limit中的骑手id=(1, 2, 4, 6, 10),其中骑手id为10的骑手在 tb_rider表中是没有的,所以 name为null。

小测验

尝试下将上面的这条Sql语句改写成LEFT JOIN吧(要求得到相同的查询结果)?

[3] 一丝不苟的INNER JOIN

之所以叫“一丝不苟”的INNER JOIN,是因为INNER JOIN是非常严格的关联查询,换句话说,必须是根据JOIN条件两张表中存在匹配记录的才作为结果集返回。让我们回顾下[1]中LEFT JOIN的Sql。

SELECT tb_rider.id, tb_grab_order_limit.order_grab_limit
FROM tb_rider LEFT JOIN tb_grab_order_limit
ON tb_rider.id = tb_grab_order_limit.rider_id;

它的返回结果是。

idorder_grab_limit1112949677\8\5\3\

如果我们将LEFT JOIN改为INNER JOIN呐?修改后的Sql像这样。

SELECT tb_rider.id, tb_grab_order_limit.order_grab_limit
FROM tb_rider INNER JOIN tb_grab_order_limit
ON tb_rider.id = tb_grab_order_limit.rider_id;

这时返回的查询结果变成了。

idorder_grab_limit111294967

这是因为INNER JOIN会遍历连接一侧的表,根据ON后的连接条件,和连接另一侧的表进行比较,只有两张表中存在匹配的记录才会作为结果集返回。例如这里,它会遍历 tb_rider表中 id字段的值,并且去 tb_grab_order_limit表中寻找 rider_id与之匹配的记录,如果找到则作为结果返回。

小测验

猜测一下下面的这条Sql语句的返回结果是什么?

SELECT tb_rider.id, tb_grab_order_limit.order_grab_limit
FROM tb_grab_order_limit INNER JOIN tb_rider
ON tb_grab_order_limit.rider_id = tb_rider.id;

提示:这里交换了一下INNER JOIN连接的两张表的位置,根据INNER JOIN的特性,查询结果会有影响嘛?

[4] 心大的FULL JOIN

FULL JOIN其实并不在乎匹配与否,而是将连接的两张表中所有的行都返回,如果有匹配的则返回匹配的结果,如果没有匹配则哪张表中缺失则对应的将当前这条记录标记为null。看一个例子就明白啦!

SELECT tb_rider.id, tb_rider.name, tb_grab_order_limit.rider_id, tb_grab_order_limit.order_grab_limit
FROM tb_rider FULL JOIN tb_grab_order_limit ON tb_rider.id = tb_grab_order_limit.rider_id;

这条Sql语句的查询结果是这样的。

idnamerider_idorder_grab_limit1Stark1112Banner294Thor496Barton673Rogers\\5Natasha\\7Coulson\\8Coulson\\\\1010

可以看到 tb_rider表中骑手id=(3, 5, 7, 8)的骑手在 tb_grab_order_limit表中没有匹配的记录,而 tb_grab_order_limit表中骑手id=(10)的骑手在 tb_rider表中没有匹配记录,但是它们都作为结果集返回了。只不过缺失 tb_grab_order_limit记录的, rider_id和 order_grab_limit字段值为null,而缺失 tb_rider记录的, id和 name字段的值为null。

事实上,绝大多数情况下,FULL JOIN都不会被用到。而且在一些数据库管理系统中,例如MySql(我们的线上环境主要使用的就是MySql),是不支持FULL JOIN语法的。对于上面的查询语句,需要使用一些技巧通过LEFT JOIN、RIGHT JOIN以及UNION(这篇教程中我们不讨论UNION语法哦)语法的组合来实现同样效果的查询。

SELECT tb_rider.id, tb_rider.name, tb_grab_order_limit.rider_id, tb_grab_order_limit.order_grab_limit
FROM tb_rider LEFT JOIN tb_grab_order_limit ON tb_rider.id = tb_grab_order_limit.rider_id
UNION
SELECT tb_rider.id, tb_rider.name, tb_grab_order_limit.rider_id, tb_grab_order_limit.rider_id
FROM tb_rider RIGHT JOIN tb_grab_order_limit ON tb_rider.id = tb_grab_order_limit.rider_id
WHERE tb_rider.id IS null;

这已经超出了这篇教程的讨论范围啦!如果想要挑战一下自己,以下是一些提示。

  1. UNION连接两条SELECT语句,作用是将两个SELECT语句的查询结果取交集;
  2. 第2条SELECT语句中的 WHERE tb_rider.id ISnull 是为了对存在匹配的数据记录去重(否则UNION之后会有重复的结果);
  3. WHERE语句是在RIGHT JOIN之后,UNION之前执行的;

试着在这两条提示下理解一下这条Sql语句,如果能够弄明白这条语句是如何等价于FULL JOIN的,那么说明你对JOIN家族的语法已经基本掌握啦。如果暂时还不能弄得非常明白也没关系,多看一看例子,多写一写实践一下,慢慢就会明白啦。

题外话

从上面的讲解我们了解到JOIN的四种用法,总结一下。

  1. INNER JOIN关键字在两张表中都有匹配的值的时候返回匹配的行;
  2. LEFT JOIN关键字从左表返回所有的行,即使在右表中没有匹配的行;
  3. RIGHT JOIN关键字从右表返回所有的行,即使在左表中没有匹配的行;
  4. FULL JOIN关键字从左表和右表那里返回所有行,即使右表的行在左表中没有匹配或者左表的行在右表中没有匹配,这些行也会返回;

不过这些都是刻板的文字总结,让我们换个视角总结一下这集中JOIN语法。

离散数学中在讨论集合论的时候介绍过“韦恩图”的概念,它清楚的描述了数据集合之间的关系。而JOIN的这4种操作也正好对应了4种集合运算,下面的这张图(Figure 1)很清楚的描述了这种关系。

4.4 嵌套的SELECT语法

再来看一下讲述LEFT JOIN的开始,我们提到的那个例子:查询 tb_rider表中所有骑手对应的自定义接单上限。当时我们首先提出了思路1,是分为2个步骤的。

SELECT id FROM tb_rider;

SELECT rider_id, order_grab_limit FROM tb_grab_order_limit
WHERE rider_id IN (1, 2, 3, 4, 5, 6, 7, 8);

我们说这个思路不好,这是显然的,因为在现实场景中往往数据集合都很大(例如这里的 rider_id在现实中可能是成百上千甚至成千上万个),思路本身没有问题但无法操作执行。所以在4.3节我们选择通过JOIN语法来实现同样的查询。那是不是思路1就真的只能是个纸上谈兵的思路了呢?当然不是啦!我们还可以使用嵌套的SELECT语句,就像这样。

SELECT rider_id, order_grab_limit FROM tb_grab_order_limit
WHERE rider_id IN (SELECT id FROM tb_rider);

这个写法非常好理解, WHERE rider_id IN(SELECT id FROM tb_rider)首先执行括号中的语句 SELECT id FROM tb_rider,然后执行IN筛选,就是我们的思路1描述的那样。于是得到下面的结果。

rider_idorder_grab_limit111294967

复习题

回想一下上面的结果和以下哪条Sql语句的执行结果是一致的呢?为什么是一致的,为什么和其他的不一致?

1. SELECT tb_rider.id, tb_grab_order_limit.order_grab_limit
 FROM tb_rider LEFT JOIN tb_grab_order_limit
 ON tb_rider.id = tb_grab_order_limit.rider_id;
2. SELECT tb_grab_order_limit.rider_id, tb_rider.name
 FROM tb_rider RIGHT JOIN tb_grab_order_limit
 ON tb_rider.id = tb_grab_order_limit.rider_id;
3. SELECT tb_rider.id, tb_grab_order_limit.order_grab_limit
 FROM tb_rider INNER JOIN tb_grab_order_limit
 ON tb_rider.id = tb_grab_order_limit.rider_id;
4. SELECT tb_rider.id, tb_grab_order_limit.order_grab_limit
 FROM tb_rider FULL JOIN tb_grab_order_limit
 ON tb_rider.id = tb_grab_order_limit.rider_id;

小测验

思考一下以下这个场景,看看能否写出它对应的Sql语句?

场景:筛选出所有通过实名认证( real_name_certify_state=2)的金牌( level=3)骑手( tb_rider表),在2017-12-30当天( created_at>=xxx AND created_at

想一想有几种写法呢?

5 闯关答题:快速复习

前面的几个段落我们学习了Sql查询中最常用,而且特别好用的语法知识,让我们简单总结一下。

  1. 数据库、数据表的概念;
  2. 最基本的Sql查询结构;
  3. IN查询和逻辑操作语法(AND/OR);
  4. 对查询结果进行排序和LIMIT语法;
  5. 聚合函数(COUNT/AVG/SUM)和DISTINCT语法;
  6. 对查询结果分组(GROUP BY);
  7. 对聚合函数的结果进行筛选的HAVING语法;
  8. 字段类型和索引的概念和作用;
  9. JOIN语法的一家子(LEFT JOIN/RIGHT JOIN/INNER JOIN/FULL JOIN);
  10. 嵌套的SELECT语法;

学习了这么多知识点,实在是太腻害了!给自己点赞!

但是(凡事都有个但是)...

想要把这些知识点融会贯通,灵活应用到现实工作中更多变、更复杂的查询场景,仅仅是“学会”是不够的,还需要更多的“练习”和“回味”。

这个部分我设计了一个“闯关答题”项目,通过思考和回答这些闯关题,帮助你更好的掌握上面提到的知识点。

先来看一下答题将要用到的数据表。

[1] 商品数据表: tb_product

idproduct_idnameprice11001iPad Pro 10.5 64G WLAN488821002Macbook Pro 2017 13.3 i5/8G/256GB1388831003iPhone X 64G8388

建表语句:

CREATE TABLE `tb_product` (
 `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '自增主键',
 `product_id` bigint(20) NOT NULL DEFAULT '0' COMMENT '商品id',
 `name` varchar(100) NOT NULL DEFAULT '' COMMENT '商品名称',
 `price` int(11) NOT NULL DEFAULT '0' COMMENT '商品价格',
 PRIMARY KEY (`id`),
 UNIQUE KEY `uk_product_id` (`product_id`)
) ENGINE=InnoDB AUTO_INCREMENT=14 DEFAULT CHARSET=utf8 COMMENT='商品信息表';

字段含义:

  1. id:自增主键;
  2. product_id:商品id;
  3. name:商品名称;
  4. price:商品单价,单位是元;

[2] 用户数据表: tb_customer


idcustomer_idnamegenderbalance1NO100001火火女188882NO100002拨泼抹女90003NO100003艾桥男79904NO100004水娃女8388

建表语句:

CREATE TABLE `tb_customer` (
 `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '自增主键',
 `customer_id` varchar(100) NOT NULL DEFAULT '' COMMENT '用户id',
 `name` varchar(100) NOT NULL DEFAULT '' COMMENT '用户姓名',
 `gender` varchar(30) NOT NULL DEFAULT '' COMMENT '用户性别',
 `balance` int(11) NOT NULL DEFAULT '0' COMMENT '账户余额',
 PRIMARY KEY (`id`),
 UNIQUE KEY `uk_customer_id` (`customer_id`)
) ENGINE=InnoDB AUTO_INCREMENT=14 DEFAULT CHARSET=utf8 COMMENT='用户信息表';

字段含义:

  1. id:自增主键;
  2. customer_id:用户id;
  3. name:用户姓名;
  4. gender:用户的性别;
  5. balance:用户当前的可用账户余额,单位是元;

[3] 订单数据表: tb_order

idorder_idcustomer_idproduct_idquantity1NUM1000301NO100001100112NUM1000302NO100001100223NUM1000303NO100002100224NUM1000304NO100003100215NUM1000305NO10000110031

建表语句:

CREATE TABLE `tb_order` (
 `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '自增主键',
 `order_id` varchar(100) NOT NULL DEFAULT '' COMMENT '订单id',
 `customer_id` varchar(100) NOT NULL DEFAULT '0' COMMENT '用户id',
 `product_id` bigint(20) NOT NULL DEFAULT '0' COMMENT '商品id',
 `quantity` int(11) NOT NULL DEFAULT '0' COMMENT '商品价格',
 PRIMARY KEY (`id`),
 UNIQUE KEY `uk_order_id` (`order_id`)
) ENGINE=InnoDB AUTO_INCREMENT=14 DEFAULT CHARSET=utf8 COMMENT='订单数据表';

字段含义:

  1. id:自增主键;
  2. order_id:订单号;
  3. customer_id:下单用户id;
  4. product_id:购买的商品id;
  5. quantity:购买的数量;

了解完需要用到表结构,我们就要开始答题啦!

第一关:查询账户余额大于1万元的用户id和姓名?

Answer:

SELECT customer_id, name FROM tb_customer WHERE balance > 10000;

customer_idnameNO100001火火

第二关:查询账户余额小于1万元且性别为女生的用户姓名?

Answer:

SELECT name FROM tb_customer WHERE balance < 10000 AND gender="女";

| name | |--------| | 拨泼抹 | | 水娃 |

第三关:查询用户id为NO100001和NO100002的用户,所有购买记录的订单号?

Hint:IN

Answer:

SELECT order_id FROM tb_order WHERE customer_id IN ("NO100001", "NO100002");

| order_id | |------------| | NUM1000301 | | NUM1000302 | | NUM1000303 | | NUM1000305 |

第四关:查询用户id为NO100001、NO100002两位用户所有的购买记录(所有字段),要求按照优先以商品id递增、其次以订单号递减的规则展示数据?

Hint:IN、ORDER BY

Answer:

SELECT * FROM tb_order WHERE customer_id IN ("NO100001", "NO100002")
ORDER BY product_id ASC, order_id DESC;

idorder_idcustomer_idproduct_idquantity1NUM1000301NO100001100113NUM1000303NO100002100222NUM1000302NO100001100225NUM1000305NO10000110031

第五关:查询性别为女生的用户总数?

Hint:COUNT

Answer:

SELECT COUNT(customer_id) FROM tb_customer WHERE gender="女";

| COUNT(customer_id) | |---------------------| | 3 |

第六关:查询NO100001、NO100002、NO100003三位用户各自购买商品的总数(不区分商品类型),输出购买商品件数大于等于2件的用户id以及他们对应购买的商品总数?

Warning:“购买商品的总数”和上一关“女生用户的总数”,这两个“总数”一样吗?

Hint:IN、SUM、HAVING

Answer:

SELECT customer_id, SUM(quantity) FROM tb_order
WHERE customer_id IN ("NO100001", "NO100002", "NO100003")
GROUP BY customer_id
HAVING SUM(quantity) >= 2;

customer_idSUM(quantity)NO1000014NO1000022

第七关:查询NO100001、NO100002、NO100003三位用户各自购买商品的总数(不区分商品类型),输出购买总数前两名的用户id以及他们对应购买的商品总数?

Hint:IN、SUM、ORDER BY、LIMIT

Answer:

SELECT customer_id, SUM(quantity) FROM tb_order
WHERE customer_id IN ("NO100001", "NO100002", "NO100003")
GROUP BY customer_id
ORDER BY SUM(quantity) DESC
LIMIT 2;

customer_idSUM(quantity)NO1000014NO1000022

第八关:查询所有用户各自购买商品的总数(不区分商品类型),输出购买商品件数大于等于2件的用户id以及他们对应购买的商品总数?要求给出至少两种写法。

Warning:注意是“所有用户”,不是所有的用户都购买了商品

Hint:关联查询有哪些方法?

Answer:

写法一:嵌套的SELECT

SELECT customer_id, SUM(quantity) FROM tb_order
WHERE customer_id IN (SELECT customer_id FROM tb_customer)
GROUP BY customer_id
HAVING SUM(quantity) >= 2;

customer_idSUM(quantity)NO1000014NO1000022

写法二:使用LEFT JOIN语法

SELECT tb_customer.customer_id, SUM(tb_order.quantity) FROM tb_customer
LEFT JOIN tb_order ON tb_customer.customer_id = tb_order.customer_id
GROUP BY tb_customer.customer_id
HAVING SUM(tb_order.quantity) >= 2;

customer_idSUM(tb_order.quantity)NO1000014NO1000022

第九关:查询所有用户各自购买商品的总数(不区分商品类型),输出购买总数前两名的用户id以及他们对应购买的商品总数?要求给出至少两种写法。

Hint:关联查询有哪些方法?

Answer:

写法一:嵌套的SELECT

SELECT customer_id, SUM(quantity) FROM tb_order
WHERE customer_id IN (SELECT customer_id FROM tb_customer)
GROUP BY customer_id
ORDER BY SUM(quantity) DESC
LIMIT 2;

customer_idSUM(quantity)NO1000014NO1000022

写法二:使用LEFT JOIN语法

SELECT tb_customer.customer_id, SUM(tb_order.quantity) FROM tb_customer
LEFT JOIN tb_order ON tb_customer.customer_id = tb_order.customer_id
GROUP BY tb_customer.customer_id
ORDER BY SUM(tb_order.quantity) DESC
LIMIT 2;

customer_idSUM(tb_order.quantity)NO1000014NO1000022

第十关:以下哪几条Sql语句使用到了索引?分别是哪些字段上的索引?是什么类型的索引?

1. SELECT name FROM tb_customer WHERE customer_id = 1001;
2. SELECT product_id, name FROM tb_product WHERE price > 5000;
3. SELECT order_id, customer_id, product_id FROM tb_order
 WHERE order_id = "NUM1000302" AND customer_id = "NO100001"
 AND product_id = "1002";
4. SELECT order_id FROM tb_order WHERE id > 2;

Hint:索引

Answer:

sql序号是否使用到索引索引所在字段索引类型1是customer_idUNIQUE KEY2否--3是order_idUNIQUE KEY4是idPRIMARY KEY


阅读博客还不过瘾?

sql数据运营(写给运营同学和初学者的 SQL 入门教程)
运营那些事儿
分享到朋友圈
收藏
收藏
评分

综合评分:

我的评分
Xinstall 15天会员特权
Xinstall是专业的数据分析服务商,帮企业追踪渠道安装来源、裂变拉新统计、广告流量指导等,广泛应用于广告效果统计、APP地推与CPS/CPA归属统计等方面。
20羽毛
立即兑换
一书一课30天会员体验卡
领30天VIP会员,110+门职场大课,250+本精读好书免费学!助你提升职场力!
20羽毛
立即兑换
顺丰同城急送全国通用20元优惠券
顺丰同城急送是顺丰推出的平均1小时送全城的即时快送服务,专业安全,准时送达!
30羽毛
立即兑换
运营那些事儿
运营那些事儿
发表文章43455
确认要消耗 羽毛购买
sql数据运营(写给运营同学和初学者的 SQL 入门教程)吗?
考虑一下
很遗憾,羽毛不足
我知道了

我们致力于提供一个高质量内容的交流平台。为落实国家互联网信息办公室“依法管网、依法办网、依法上网”的要求,为完善跟帖评论自律管理,为了保护用户创造的内容、维护开放、真实、专业的平台氛围,我们团队将依据本公约中的条款对注册用户和发布在本平台的内容进行管理。平台鼓励用户创作、发布优质内容,同时也将采取必要措施管理违法、侵权或有其他不良影响的网络信息。


一、根据《网络信息内容生态治理规定》《中华人民共和国未成年人保护法》等法律法规,对以下违法、不良信息或存在危害的行为进行处理。
1. 违反法律法规的信息,主要表现为:
    1)反对宪法所确定的基本原则;
    2)危害国家安全,泄露国家秘密,颠覆国家政权,破坏国家统一,损害国家荣誉和利益;
    3)侮辱、滥用英烈形象,歪曲、丑化、亵渎、否定英雄烈士事迹和精神,以侮辱、诽谤或者其他方式侵害英雄烈士的姓名、肖像、名誉、荣誉;
    4)宣扬恐怖主义、极端主义或者煽动实施恐怖活动、极端主义活动;
    5)煽动民族仇恨、民族歧视,破坏民族团结;
    6)破坏国家宗教政策,宣扬邪教和封建迷信;
    7)散布谣言,扰乱社会秩序,破坏社会稳定;
    8)宣扬淫秽、色情、赌博、暴力、凶杀、恐怖或者教唆犯罪;
    9)煽动非法集会、结社、游行、示威、聚众扰乱社会秩序;
    10)侮辱或者诽谤他人,侵害他人名誉、隐私和其他合法权益;
    11)通过网络以文字、图片、音视频等形式,对未成年人实施侮辱、诽谤、威胁或者恶意损害未成年人形象进行网络欺凌的;
    12)危害未成年人身心健康的;
    13)含有法律、行政法规禁止的其他内容;


2. 不友善:不尊重用户及其所贡献内容的信息或行为。主要表现为:
    1)轻蔑:贬低、轻视他人及其劳动成果;
    2)诽谤:捏造、散布虚假事实,损害他人名誉;
    3)嘲讽:以比喻、夸张、侮辱性的手法对他人或其行为进行揭露或描述,以此来激怒他人;
    4)挑衅:以不友好的方式激怒他人,意图使对方对自己的言论作出回应,蓄意制造事端;
    5)羞辱:贬低他人的能力、行为、生理或身份特征,让对方难堪;
    6)谩骂:以不文明的语言对他人进行负面评价;
    7)歧视:煽动人群歧视、地域歧视等,针对他人的民族、种族、宗教、性取向、性别、年龄、地域、生理特征等身份或者归类的攻击;
    8)威胁:许诺以不良的后果来迫使他人服从自己的意志;


3. 发布垃圾广告信息:以推广曝光为目的,发布影响用户体验、扰乱本网站秩序的内容,或进行相关行为。主要表现为:
    1)多次发布包含售卖产品、提供服务、宣传推广内容的垃圾广告。包括但不限于以下几种形式:
    2)单个帐号多次发布包含垃圾广告的内容;
    3)多个广告帐号互相配合发布、传播包含垃圾广告的内容;
    4)多次发布包含欺骗性外链的内容,如未注明的淘宝客链接、跳转网站等,诱骗用户点击链接
    5)发布大量包含推广链接、产品、品牌等内容获取搜索引擎中的不正当曝光;
    6)购买或出售帐号之间虚假地互动,发布干扰网站秩序的推广内容及相关交易。
    7)发布包含欺骗性的恶意营销内容,如通过伪造经历、冒充他人等方式进行恶意营销;
    8)使用特殊符号、图片等方式规避垃圾广告内容审核的广告内容。


4. 色情低俗信息,主要表现为:
    1)包含自己或他人性经验的细节描述或露骨的感受描述;
    2)涉及色情段子、两性笑话的低俗内容;
    3)配图、头图中包含庸俗或挑逗性图片的内容;
    4)带有性暗示、性挑逗等易使人产生性联想;
    5)展现血腥、惊悚、残忍等致人身心不适;
    6)炒作绯闻、丑闻、劣迹等;
    7)宣扬低俗、庸俗、媚俗内容。


5. 不实信息,主要表现为:
    1)可能存在事实性错误或者造谣等内容;
    2)存在事实夸大、伪造虚假经历等误导他人的内容;
    3)伪造身份、冒充他人,通过头像、用户名等个人信息暗示自己具有特定身份,或与特定机构或个人存在关联。


6. 传播封建迷信,主要表现为:
    1)找人算命、测字、占卜、解梦、化解厄运、使用迷信方式治病;
    2)求推荐算命看相大师;
    3)针对具体风水等问题进行求助或咨询;
    4)问自己或他人的八字、六爻、星盘、手相、面相、五行缺失,包括通过占卜方法问婚姻、前程、运势,东西宠物丢了能不能找回、取名改名等;


7. 文章标题党,主要表现为:
    1)以各种夸张、猎奇、不合常理的表现手法等行为来诱导用户;
    2)内容与标题之间存在严重不实或者原意扭曲;
    3)使用夸张标题,内容与标题严重不符的。


8.「饭圈」乱象行为,主要表现为:
    1)诱导未成年人应援集资、高额消费、投票打榜
    2)粉丝互撕谩骂、拉踩引战、造谣攻击、人肉搜索、侵犯隐私
    3)鼓动「饭圈」粉丝攀比炫富、奢靡享乐等行为
    4)以号召粉丝、雇用网络水军、「养号」形式刷量控评等行为
    5)通过「蹭热点」、制造话题等形式干扰舆论,影响传播秩序


9. 其他危害行为或内容,主要表现为:
    1)可能引发未成年人模仿不安全行为和违反社会公德行为、诱导未成年人不良嗜好影响未成年人身心健康的;
    2)不当评述自然灾害、重大事故等灾难的;
    3)美化、粉饰侵略战争行为的;
    4)法律、行政法规禁止,或可能对网络生态造成不良影响的其他内容。


二、违规处罚
本网站通过主动发现和接受用户举报两种方式收集违规行为信息。所有有意的降低内容质量、伤害平台氛围及欺凌未成年人或危害未成年人身心健康的行为都是不能容忍的。
当一个用户发布违规内容时,本网站将依据相关用户违规情节严重程度,对帐号进行禁言 1 天、7 天、15 天直至永久禁言或封停账号的处罚。当涉及欺凌未成年人、危害未成年人身心健康、通过作弊手段注册、使用帐号,或者滥用多个帐号发布违规内容时,本网站将加重处罚。


三、申诉
随着平台管理经验的不断丰富,本网站出于维护本网站氛围和秩序的目的,将不断完善本公约。
如果本网站用户对本网站基于本公约规定做出的处理有异议,可以通过「建议反馈」功能向本网站进行反馈。
(规则的最终解释权归属本网站所有)

我知道了
恭喜你~答对了
+5羽毛
下一次认真读哦
成功推荐给其他人
+ 10羽毛
评论成功且进入审核!审核通过后,您将获得10羽毛的奖励。分享本文章给好友阅读最高再得15羽毛~
(羽毛可至 "羽毛精选" 兑换礼品)
好友微信扫一扫
复制链接