表的基本查询
- 表的增删查改
-
- [1. Create](#1. Create)
- [2. Retrieve](#2. Retrieve)
- [3. Update](#3. Update)
- [4. Delete](#4. Delete)
- [5. 插入查询结果](#5. 插入查询结果)
- [6. 聚合函数](#6. 聚合函数)
- [7. group by 子句的使用](#7. group by 子句的使用)
- [8. 相关题目练习](#8. 相关题目练习)
表的增删查改
表的增删查改,简称表的 CURD 操作 : Create(创建) ,Update(更新) ,Retrieve(读取) ,Delete(删除).
下面我们逐一进行介绍。
1. Create
语法:
INSERT [INTO] table_name
[(column [, column] ...)]
VALUES (value_list) [, (value_list)] ...
value_list: value, [, value] ...
例如创建一张学生表:
create table students(
-> id int unsigned primary key auto_increment,
-> stunum int not null unique comment '学号',
-> name varchar(20) not null,
-> email varchar(20)
-> );
(1)单行数据 + 全列插入
接下来我们插入两条记录,其中 value_list 数量必须和定义表的列的数量及顺序一致:
例如插入一个数据:
insert into students values(1, 1001, 'Smith', '11111');
其中 into 可以省略;示例如下:
data:image/s3,"s3://crabby-images/d60ab/d60ab304162d6d9f6742a59eea245dc3f00a1ade" alt=""
(2)多行数据 + 指定列插入
我们可以在表的名字后面带上需要插入的字段,并且 value_list 数量必须和指定字段数量及顺序一致,例如,我们只想插入 stunum 和name 字段:
insert into students(stunum, name) values(1003, 'Mike');
示例如下:
data:image/s3,"s3://crabby-images/594b4/594b4b1427ef5b3c18e738dc5006851ca60d7d2c" alt=""
(3)插入否则更新
由于主键 或者唯一键对应的值已经存在会而导致插入失败。
我们先查看我们当前表的数据:
data:image/s3,"s3://crabby-images/bd0cf/bd0cfe10235edb8909436bc72a23cab42c2785a8" alt=""
其中 id 为主键,stunum 为唯一键,所以我们分别尝试插入主键和唯一键冲突的数据:
data:image/s3,"s3://crabby-images/027aa/027aa3361048a1e5e6137531673eb5566d892c8d" alt=""
如上图,插入失败。所以我们可以选择性的进行同步更新操作语法:
insert...
on duplicate key update column = value [, column = value] ...
例如我们想插入 Jane 这位同学的数据,但是我们并不知道这位同学是否存在于这张表中,所以我们使用上面的语法,假设不存在,就按照我们的数据插入数据;否则我们将她的 stunum 更新为 1010:
insert into students(id, stunum, name) values(2, 1010, 'Jane')
-> on duplicate key update stunum = 1010, name = 'Jane';
如下:
data:image/s3,"s3://crabby-images/71c24/71c24e93821044890ca59ace14f0c2a7be111359" alt=""
我们可以观察到当插入成功后,会有 2 rows affected (0.00 sec) 这样的提示,即:
data:image/s3,"s3://crabby-images/34750/34750edf1b32ea642a2cc1a93f6987637e837917" alt=""
这个提示的含义如下:
- 0 row affected: 表中有冲突数据,但冲突数据的值和 update 的值相等
- 1 row affected: 表中没有冲突数据,数据被插入
- 2 row affected: 表中有冲突数据,并且数据已经被更新
(4)替换
替换的本质:主键或者唯一键没有冲突,则直接插入;主键 或者 唯一键 如果冲突,则删除后再插入。
替换将 insert 换成 replace 使用即可。示例如下:
data:image/s3,"s3://crabby-images/a1fd0/a1fd02910e39c422e0bab203f4d1dc31da0cad6c" alt=""
2. Retrieve
Retrieve 指的是表的读取。
语法:
SELECT
[DISTINCT] {* | {column [, column] ...}
[FROM table_name]
[WHERE ...]
[ORDER BY column [ASC | DESC], ...]
LIMIT ...
我们先创建一个表结构:
create table exam_result(
-> id int unsigned primary key auto_increment,
-> name varchar(20) not null comment '姓名',
-> chinese float default 0.0 comment '语文成绩',
-> math float default 0.0 comment '数学成绩',
-> english float default 0.0 comment '英语成绩'
-> );
data:image/s3,"s3://crabby-images/89dfc/89dfce6f5aca281c3be350336bdd25448591450a" alt=""
接下来插入一些数据:
insert into exam_result(name, chinese, math, english) values
-> ('Smith', 67, 98, 62),
-> ('Jane', 64, 78, 79),
-> ('Tiny', 76, 87, 77),
-> ('John', 89, 78, 90),
-> ('Mike', 90, 91, 93),
-> ('Lily', 82, 83, 89),
-> ('Klin', 60, 53, 59);
data:image/s3,"s3://crabby-images/05b32/05b32e25b822053f80f77842ebdba9984b20b19a" alt=""
(1)select 列
a. 全列查询
语法:SELECT * FROM 表名;
通常情况下不建议使用 * 进行全列查询,因为:
-
- 查询的列越多,意味着需要传输的数据量越大;
-
- 可能会影响到索引的使用。(索引待后面讲解)
例如:
data:image/s3,"s3://crabby-images/4290f/4290ff29b70e386b104036bfd244234e4ad4a599" alt=""
b. 指定列查询
指定列的顺序不需要按定义表的顺序来,语法就是在 select 后跟上指定的字段列即可。例如我们只需要查询 id 、name 、math,如下:
select id, name, math from exam_result;
data:image/s3,"s3://crabby-images/75fc9/75fc908944993d749f54dc27a6884b909992fbe3" alt=""
c. 查询字段为表达式
- 表达式不包含字段:
select id, name, 10 from exam_result;
- - - 显示 10
data:image/s3,"s3://crabby-images/b982d/b982df27bc39008c5f68a5cfd89c9267218d8774" alt=""
- 表达式包含一个字段:
select id, name, chinese+10 from exam_result;
- - - 使语文成绩 + 10 分
data:image/s3,"s3://crabby-images/d1420/d1420e74d627beb14ce3e0e0145b57aad711d75d" alt=""
- 表达式包含多个字段:
select id, name, chinese + math + english from exam_result;
- - - 统计总成绩
data:image/s3,"s3://crabby-images/2ff67/2ff677cb226fa89b559e56357e2810eb761dad96" alt=""
d. 为查询结果指定别名
语法:SELECT column [AS] alias_name [...] FROM table_name;
例如将上面统计的总成绩的字段改为总分:select id, name, chinese + math + english as 总分 from exam_result;
- - - as 可以加也可以不加
data:image/s3,"s3://crabby-images/47292/47292aed16d8194681ff35d3a66c21bcbc48264c" alt=""
e. 结果去重
去重使用关键字 DISTINCT ,直接加在 select 后即可。例如 math 中 78 的分数重复了:
data:image/s3,"s3://crabby-images/6f5f1/6f5f13359aa6d9bab5b4229afbb2a11c0dc7b73b" alt=""
结果去重:select distinct math from exam_result;
data:image/s3,"s3://crabby-images/ec948/ec9482efe5db9e50404713eb2de9673797856599" alt=""
(2)where 条件
- 比较运算符:
data:image/s3,"s3://crabby-images/f20fc/f20fcf0e64577cce3003b028a60f28e313e5c600" alt=""
- 逻辑运算符:
data:image/s3,"s3://crabby-images/815f6/815f6ad68a953d899ec1aeb8b0eac103745dc350" alt=""
使用 where 筛选条件只需将 where 放在语句最后,后面跟上条件即可,下面举一些实例:
-
英语不及格的同学及英语成绩 ( < 60 )
select name, english from exam_result where english < 60;
data:image/s3,"s3://crabby-images/09294/092940a53cb24c5a0a3980f4a430595071a8aa7a" alt=""
- 语文成绩在 [80, 90] 分的同学及语文成绩
-
使用 and 进行条件连接
select name, chinese from exam_result where chinese >= 80 and chinese <= 90;
data:image/s3,"s3://crabby-images/6ef95/6ef95f541bb631199fd49e3f2a445938ecfb2f95" alt=""
-
使用 between ... and ... 条件
select name, chinese from exam_result where chinese between 80 and 90;
data:image/s3,"s3://crabby-images/3ed56/3ed56bccdcf68628dc8a071381ee388f0b3e05b2" alt=""
- 数学成绩是 53 或者 91 或者 98 分的同学及数学成绩
-
使用 or 进行条件连接
select name, math from exam_result -> where math = 53 -> or math = 91 -> or math = 98;
data:image/s3,"s3://crabby-images/48c7a/48c7a6847d7d352db8c71761f41df328a329e0fa" alt=""
-
使用 in 条件
select name, math from exam_result where math in(53, 91, 98);
data:image/s3,"s3://crabby-images/20f5f/20f5f96b6ac664b6ac42c713fee5e5ff121ffd94" alt=""
- 以字母 J 开头的同学
-
% 匹配任意多个(包括 0 个)任意字符
select name from exam_result where name like 'J%';
data:image/s3,"s3://crabby-images/67e40/67e40f1f81e0eed376b2c4477dad2c9f35b5efab" alt=""
- _ 匹配严格的一个任意字符
例如需要查找 J 某同学:
select name from exam_result where name like 'J_';
- 英语成绩好于数学成绩的同学
-
where 条件中比较运算符两侧都是字段
select name, math, english from exam_result where english > math;
data:image/s3,"s3://crabby-images/80816/80816ed79fa08e85974fb4e32f662c2d87db8f92" alt=""
- 总分在 200 分以下的同学
-
where 条件中使用表达式
-
别名不能用在 where 条件中,因为有 where 语句的语句中,先筛选 where 语句的条件,再读取筛选后的表
select name, chinese+math+english total from exam_result where chinese+math+english < 200;
data:image/s3,"s3://crabby-images/6985f/6985f2eb597fd00cab60eaf67229120a2aae7a7d" alt=""
- 英语成绩 > 80 并且姓名首字母不以 M 开头的同学
-
and 于 not 的使用
select name, english from exam_result -> where english > 80 and name not like 'M%';
data:image/s3,"s3://crabby-images/cf121/cf121e68c790d3a50d5ba8bca2e584665604f850" alt=""
- 名字首字母以 T 开头的同学,否则要求总成绩 > 200 并且 语文成绩 < 数学成绩 并且 英语成绩 > 80
-
综合性查询
select name, chinese, math, english, chinese+math+english total from exam_result -> where name like 'T%' -> or chinese+math+english > 200 -> and chinese < math -> and english > 80;
data:image/s3,"s3://crabby-images/12f4b/12f4b77d0ce4da559b0f21dc0fc04892f368385e" alt=""
- null 的查询
接下来我们切回 students 表进行查询:
data:image/s3,"s3://crabby-images/c4f20/c4f20fe00fce7ecec1b4ac03d068fd3c466be7c5" alt=""
-
查询 email 已知的同学姓名
select name, email from students where email is not null;
data:image/s3,"s3://crabby-images/e3685/e3685f09e7c90f6579d225b36842cbde89e2435d" alt=""
-
NULL 和 NULL 的比较,= 和 <=> 的区别
select null = null, null = 1, null = 0;
data:image/s3,"s3://crabby-images/90192/901925d10f243e38195c755e2a036034b3bf7004" alt=""
select NULL <=> NULL, NULL <=> 1, NULL <=> 0
data:image/s3,"s3://crabby-images/a3ad6/a3ad6cae1a2a900908f20fd15daa9eae126dcc3d" alt=""
(3)结果排序
语法:
-
asc 为升序(从小到大)
-
desc 为降序(从大到小)
-
默认为 asc
select ... from table_name [where ...] order by column [asc|desc], [...];
注意:没有 order by 子句的查询,返回的顺序是未定义的,永远不要依赖这个顺序。
下面看实例:
-
同学及数学成绩,按数学成绩升序显示
select name, math from exam_result order by math asc;
注:asc 可以省略
data:image/s3,"s3://crabby-images/9e903/9e9039ea4de24826d19e1b2fef131984f56e0ed3" alt=""
-
同学及 email,按 email 排序显示
select name, email from students order by email asc;
data:image/s3,"s3://crabby-images/f18c1/f18c1a4281ee7abf36bbc9cc755adb3dd8e1ad05" alt=""
注意:NULL 视为比任何值都小,升序出现在最上面。
-
查询同学各门成绩,依次按 数学降序,英语升序,语文升序的方式显示
select name, math, english, chinese from exam_result order by math desc, english asc, chinese asc;
多字段排序,排序优先级随书写顺序:
data:image/s3,"s3://crabby-images/e2982/e2982ae611cc01e6cabe38c6012d780faa279582" alt=""
- 查询同学及总分,由高到低
order by 子句中可以使用列别名,因为 order by 子句是最后执行的,要先有合适的数据,才能排序。
select name, chinese+math+english total from exam_result order by total desc;
data:image/s3,"s3://crabby-images/05edb/05edba139f6d269c483072a68716b071fc6e20aa" alt=""
- 查询姓名首字母为 J 的同学或者为 S 的同学数学成绩,结果按数学成绩由高到低显示
结合 where 子句 和 order by 子句:
select name, math from exam_result
-> where name like 'J%' or name like 'S%'
-> order by math desc;
data:image/s3,"s3://crabby-images/6442e/6442e8300fd92b4cadc0a6afcbb2c4d4c6164951" alt=""
(4)筛选分页结果
语法:
起始下标为 0
从 0 开始,筛选 n 条结果
select... from table_name [where ...] [order by ...] limit n;
从 s 开始,筛选 n 条结果
select... from table_name [where ...] [order by ...] limit s, n;
从 s 开始,筛选 n 条结果,比第二种用法更明确,建议使用
select... from table_name [where ...] [order by ...] limit n offset s;
建议:对未知表进行查询时,最好加一条 limit 1,避免因为表中数据过大,查询全表数据导致数据库卡死。
实例:
- 按 id 进行分页,每页 3 条记录,分别显示 第 1、2、3 页
第一页:
select name, chinese, math, english from exam_result
-> order by id limit 3 offset 0;
data:image/s3,"s3://crabby-images/ff973/ff973c20725eb905d9f199d990cb88350cd7bdbb" alt=""
第二页:
select name, chinese, math, english from exam_result order by id limit 3 offset 3;
data:image/s3,"s3://crabby-images/f506a/f506a3dfc3168efab62dacbd007bbb92d7675ef4" alt=""
第三页,如果结果不足 3 个,不会有影响:
select name, chinese, math, english from exam_result order by id limit 3 offset 6;
data:image/s3,"s3://crabby-images/afdd5/afdd52df02e1d4d8d50bc0229d226fee0a024da7" alt=""
3. Update
语法:
update table_name set column = expr [, column = expr ...]
[where ...] [order by ...] [limit ...];
实例:
- 将 Jane 同学的数学成绩变更为 80 分
查看原数据:
data:image/s3,"s3://crabby-images/1a443/1a44331097147a9a0b3698261ee7c2bfaba58306" alt=""
更新数据:
update exam_result set math=80 where name='Jane';
查看更新后数据:
data:image/s3,"s3://crabby-images/0a2a9/0a2a9d2d1a605b24804904e68e778870c3cbf5b8" alt=""
- 将 John 同学的语文成绩变更为 80 分,英语成绩变更为 70 分
- 一次更新多个列
查看原数据:
data:image/s3,"s3://crabby-images/17e0a/17e0a4cf9448d96869fe409f64fc5e0379db0d62" alt=""
更新数据:
update exam_result set chinese=80, english=70 where name='John';
查看更新后数据:
data:image/s3,"s3://crabby-images/60e97/60e97262f2becdfdf6c4f7ed1184666e4be00f59" alt=""
- 将总成绩倒数前三的 3 位同学的数学成绩加上 30 分
查看原数据:
select name, math, chinese+math+english total from exam_result order by total limit 3;
data:image/s3,"s3://crabby-images/eeb09/eeb09be51b180c40e53fb9428614b95087ef83c4" alt=""
更新数据:
update exam_result set math = math + 30 order by chinese+math+english limit 3;
查看更新后的数据,注意不能再用查看总分倒数前三的方式,因为给他们加上 30 分之后,他们就有可能不是倒数前三了,要单独去查看他们三个人的成绩:
select name, math, chinese+math+english total from exam_result where name in('Klin', 'Jane', 'Smith');
data:image/s3,"s3://crabby-images/5f9e7/5f9e7aaca13c2cc9d0dc30e68ff34f410fdfe0b9" alt=""
- 将所有同学的英语成绩更新为原来的 2 倍
查看原数据:
data:image/s3,"s3://crabby-images/2bb3c/2bb3c1a387fad28311aafffe21edf3e23d30ad6d" alt=""
更新数据:
update exam_result set english = english * 2;
查看更新后的数据:
data:image/s3,"s3://crabby-images/2e25a/2e25a1d05ea3c878ae62bd51aef19cc4fb9bc1a7" alt=""
注意:更新全表的语句慎用!
4. Delete
(1)删除数据
语法:
delete from table_name [where ...] [order by ...] [limit ...];
实例:
- 删除 Lily 同学的考试成绩
查看原数据:
data:image/s3,"s3://crabby-images/77ba7/77ba753ab0e8de1f6907f91c92f17ec09514f8d1" alt=""
删除数据:
delete from exam_result where name='Lily';
查看删除后的数据:
data:image/s3,"s3://crabby-images/c4fea/c4fea5f8eedec3f024a6e325efb82d9161bb4762" alt=""
- 删除整张表数据
我们查看当前库的表:
data:image/s3,"s3://crabby-images/dfcdf/dfcdf86094c0ad15d2a570827d160e7d1ae515d2" alt=""
可以看到有一张 employee 表,我们查看这张表的数据如下:
data:image/s3,"s3://crabby-images/7344e/7344e45ab31c69985764c6b80fac221a9ff9bfa6" alt=""
再查看表的结构:
data:image/s3,"s3://crabby-images/3b7d6/3b7d67cf917a91e7101102b4be35adfb8dd81832" alt=""
然后我们把整表数据删除:
delete from employee;
再查看就发现数据全没了:
data:image/s3,"s3://crabby-images/5080d/5080de35b31391d020d1aab80edbf78c178f4e8c" alt=""
再插入一条数据,自增 id 在原值上继续增长:
data:image/s3,"s3://crabby-images/43c57/43c57f0dfa24666cfee6539444aa3b3cdaa9ec44" alt=""
注意:删除整表操作要慎用!
(2)截断表
语法:
truncate [table] table_name
注意:这个操作慎用
- 只能对整表操作,不能像 delete 一样针对部分数据操作;
- 实际上 MySQL 不对数据操作,所以比 delete 更快,但是 truncate 在删除数据的时候,并不经过真正的事务,所以无法回滚;
- 会重置 auto_increme 项
接下来我们重新向 employee 表重新插入数据:
data:image/s3,"s3://crabby-images/1b8e2/1b8e294f919186b47b2677f9453a2aa8177150e6" alt=""
截断整表数据,注意影响行数是 0,所以实际上没有对数据真正操作:
truncate employee;
data:image/s3,"s3://crabby-images/842e3/842e3a972d5ffa28085b063bf0111cc25393f7df" alt=""
再插入一条数据,自增 id 在重新增长:
data:image/s3,"s3://crabby-images/070e9/070e99168ebedf5e78cf978b4dfbd4acab8c52b3" alt=""
5. 插入查询结果
语法:
insert into table_name [(column [, column ...])] select ...
实例:创建一个新表,插入一个旧表中查询到的去重后的数据
-
先创建一个旧表,并插入数据:
mysql> create table duplicate_table (id int, name varchar(20)); mysql> INSERT INTO duplicate_table VALUES -> (100, 'aaa'), -> (100, 'aaa'), -> (200, 'bbb'), -> (200, 'bbb'), -> (200, 'bbb'), -> (300, 'ccc');
-
创建一个新表,和旧表的结构一样
create table no_duplicate_table like duplicate_table;
-
将旧表的去重数据插入到新表中
insert into no_duplicate_table select distinct * from duplicate_table;
-
最后查看新表的数据
data:image/s3,"s3://crabby-images/5dc9f/5dc9f92a59b676a483f2a462ed32a243d50083e8" alt=""
6. 聚合函数
data:image/s3,"s3://crabby-images/e6ffd/e6ffd806f682f3ce9de691683da38435e42cb14d" alt=""
实例:
-
统计班级共有多少同学
select count(*) from students;
- 使用 * 做统计,不受 NULL 影响
data:image/s3,"s3://crabby-images/822c0/822c0400a1c7c5fe0eb10ad1b9242b41eb3c1555" alt=""
-
统计班级收集的 email 有多少
select count(email) from students;
- NULL 不会计入结果
data:image/s3,"s3://crabby-images/84a65/84a651d6a605e1773b66134b3f477da7d69f3dc3" alt=""
-
统计本次考试的英语成绩分数个数
select count(english) from exam_result;
- count(english) 统计的是全部成绩
data:image/s3,"s3://crabby-images/f8592/f8592cb32245fc7e5719e2b0376fa2d1477f698d" alt=""
- count(distinct math) 统计的是去重成绩数量
data:image/s3,"s3://crabby-images/34451/34451d3768d8a4faeab3c95278f93e0acee62b25" alt=""
-
统计数学成绩总分
select sum(math) from exam_result;
data:image/s3,"s3://crabby-images/50cdc/50cdc53e5df781308ddb589f9656a80167fadc01" alt=""
-
统计平均总分
select avg(chinese+math+english) 平均总分 from exam_result;
data:image/s3,"s3://crabby-images/f8718/f8718099ae74c6c4d2398beeefc1ea774a8b90fe" alt=""
-
返回语文最高分
select max(chinese) from exam_result;
data:image/s3,"s3://crabby-images/059e2/059e2b7884cc09191d5e5bde4b6ab2cf27e4a27a" alt=""
-
返回 > 70 分以上的语文最低分
select min(chinese) from exam_result where chinese > 70;
data:image/s3,"s3://crabby-images/bd548/bd5487554a62dec2d9c932a0f46a19f1df661230" alt=""
7. group by 子句的使用
在 select 中使用 group by 子句可以对指定列进行分组查询。
语法:
select column1, column2, .. from table group by column;
分组的目的是为了分组之后,方便进行聚合统计。分组的本质就是把一组按照条件拆分成了多个组,进行各自组内的统计!分组其实也是"分表",就是把一张表按照条件在逻辑上拆分成了多个子表,然后分别对各自的子表进行聚合统计!
实例,需要准备一张雇员信息表(来自oracle 9i的经典测试表) ,表的链接:oracle 9i的经典测试表包括:
- emp 员工表
- dept 部门表
- salgrade 工资等级表
查看 emp 表数据:
data:image/s3,"s3://crabby-images/84821/84821693e7ce6799cb58cfb844029db1082471e5" alt=""
查看 dept 表的数据:
data:image/s3,"s3://crabby-images/bfdea/bfdea14ed1c967aa36950f6bbc105856ec336b80" alt=""
查看 salgrade 表的数据:
data:image/s3,"s3://crabby-images/9b22d/9b22d783e3d36660f307b3eb077f8dced198b6de" alt=""
接下来开始使用分组操作:
- 显示每个部门的平均工资和最高工资
很明显,显示每个部门的平均工资和最高工资,是需要按部门分组,分别显示部门号、平均工资和最高工资:
select deptno, avg(sal), max(sal) from emp group by deptno;
data:image/s3,"s3://crabby-images/01a9f/01a9f865dbb3135ca5fe05dff4f294d2a573a19d" alt=""
- 显示每个部门的每种岗位的平均工资和最低工资
每个部门的每种岗位,即要按照部门和岗位进行分组:
select deptno, job, avg(sal), min(sal) from emp group by deptno, job;
data:image/s3,"s3://crabby-images/ce23e/ce23e03e56a2eebabb2ac44a30b27b83a453fb4e" alt=""
- 显示平均工资低于2000的部门和它的平均工资
-
先统计各个部门的平均工资
select deptno, avg(sal) from emp group by deptno;
data:image/s3,"s3://crabby-images/090d5/090d57ce4055e22013ada9f56717ca17389bd1b9" alt=""
-
使用 having 对分组进行筛选
select deptno, avg(sal) from emp group by deptno having avg(sal) < 2000;
data:image/s3,"s3://crabby-images/755cf/755cfa06d018cdc5815fe038d15b32b9591a83d8" alt=""
注意:having 和 group by 配合使用,对 group by 结果进行过滤,having 经常和 group by 搭配使用,作用是对分组进行筛选,作用有些像 where.
having 是对聚合后的统计数据进行条件筛选。其中 having 和 where 的区别在于:
- 执行的顺序不一样,where 是对任意列进行条件筛选,一般是最先执行;having 是对分组聚合后的结果进行条件筛选,一般是最后才执行!
8. 相关题目练习
注意:SQL 查询中各个关键字的执行先后顺序:
from > on> join > where > group by > with > having > select > distinct > order by > limit