高级查询

数据库示例:

学生信息表student

+-----+------+-----+-----+
| id  | name | age | sex |
+-----+------+-----+-----+
| 001 | 张三 |  18 | 男  |
| 002 | 李四 |  20 | 女  |
| 003 | 王五 |  18 | 男  |
+-----+------+-----+-----+

成绩表score

+----+------------+------------+-------+
| id | student_id | subject_id | score |
+----+------------+------------+-------+
|  1 | 001        | 1001       |    80 |
|  2 | 002        | 1002       |    60 |
|  3 | 001        | 1001       |    70 |
|  4 | 002        | 1002       |  60.5 |
+----+------------+------------+-------+

一、子查询

一个内层查询语句(select-from-where)块可以嵌套在另外一个外层查询块的where子句中,其中外层查询也称为父查询,主查询。内层查询也称子查询,从查询。
例如查询张三的各个科目的成绩:

mysql> SELECT subject_id, score FROM score WHERE student_id = (SELECT id FROM student WHERE name = '张三');
+------------+-------+
| subject_id | score |
+------------+-------+
| 1001       |    80 |
| 1002       |    70 |
+------------+-------+
2 rows in set (0.00 sec)

子查询可以较方便地对两个或多个表进行查询,过程也比较好理解。但是当查询的表过多(超过3个)时嵌套的查询就比较复杂,会可读性。而且外部的查询的返回结果不能包括内部查询的结果。

二、联结查询

联结查询可以将表进行关联,从而显示出来自多张表的数据。

1. 内联结

返回两个或者多个表之间相等关系的数据,从数学关系来看,相当于求交集。从左表中取出每一条记录,去右表中与所有的记录进行匹配:匹配必须是某个条件在左表中与右表中相同最终才会保留结果,否则不保留。

mysql>  SELECT * FROM student INNER JOIN score ON student.id = score.student_id;
+-----+------+-----+-----+----+------------+------------+-------+
| id  | name | age | sex | id | student_id | subject_id | score |
+-----+------+-----+-----+----+------------+------------+-------+
| 001 | 张三 |  18 | 男  |  1 | 001        | 1001       |    80 |
| 002 | 李四 |  20 | 女  |  2 | 002        | 1001       |    60 |
| 001 | 张三 |  18 | 男  |  3 | 001        | 1002       |    70 |
| 002 | 李四 |  20 | 女  |  4 | 002        | 1002       |  60.5 |
+-----+------+-----+-----+----+------------+------------+-------+
4 rows in set (0.00 sec)

注:表名太长时可以使用别名来区分同名字段;可以使用where代替on关键字,但效率差很多。

2. 外联结

以某张表为主,取出里面的所有记录,然后每条与另外一张表进行连接:不管能不能匹配上条件,最终都会保留:能匹配,正确保留;不能匹配,其他表的字段都置空NULL。

  • 左外联结:以左边为主表,返回左表中的所有数据:
mysql> SELECT * FROM student LEFT JOIN score ON student.id = score.student_id;
+-----+------+-----+-----+------+------------+------------+-------+
| id  | name | age | sex | id   | student_id | subject_id | score |
+-----+------+-----+-----+------+------------+------------+-------+
| 001 | 张三 |  18 | 男  |    1 | 001        | 1001       |    80 |
| 002 | 李四 |  20 | 女  |    2 | 002        | 1001       |    60 |
| 001 | 张三 |  18 | 男  |    3 | 001        | 1002       |    70 |
| 002 | 李四 |  20 | 女  |    4 | 002        | 1002       |  60.5 |
| 003 | 王五 |  18 | 男  | NULL | NULL       | NULL       |  NULL |
+-----+------+-----+-----+------+------------+------------+-------+
5 rows in set (0.00 sec)
  • 右外联结:以右边为主表,返回右表中的所有数据:
mysql> SELECT * FROM student RIGHT JOIN score ON student.id = score.student_id;
+------+------+------+------+----+------------+------------+-------+
| id   | name | age  | sex  | id | student_id | subject_id | score |
+------+------+------+------+----+------------+------------+-------+
| 001  | 张三 |   18 | 男   |  1 | 001        | 1001       |    80 |
| 001  | 张三 |   18 | 男   |  3 | 001        | 1002       |    70 |
| 002  | 李四 |   20 | 女   |  2 | 002        | 1001       |    60 |
| 002  | 李四 |   20 | 女   |  4 | 002        | 1002       |  60.5 |
+------+------+------+------+----+------------+------------+-------+
4 rows in set (0.00 sec)

3. 交叉联结

从一张表中循环取出每一条记录,每条记录都去另外一张表进行匹配:匹配一定保留(没有条件匹配),而连接本身字段就会增加(保留),最终形成的结果叫做:笛卡尔积。

SELECT * FROM table1 AS t1 CROSS JOIN table2 AS t2;

4. 自联结

有时需要在同一张表中进行联结条件的匹配或字段比较,可以使用自联结。

SELECT * FROM table t1, table t2 WHERE t1.column1=t2.column2;

三、组合查询

多数SQL查询都只包含从一个或多个表中返回数据的单条SELECT语句。MySQL也允许执行多个查询(多条SELECT语句),并将结果作为单个查询结果集返回。一般在以下两种情况中使用:

  • 在单个查询中,从不同表中返回类似结构的数据。
  • 对单个表执行多个查询,按照单个查询返回数据。
SELECT * FROM table1 WHERE condition 
UNION
SELECT * FROM table2 WHERE condition;

第一种情况,例:

mysql> SELECT student_id,score FROM score WHERE score > 65 
    -> UNION 
    -> SELECT id,age FROM student WHERE age >= 20;
+------------+-------+
| student_id | score |
+------------+-------+
| 001        |    80 |
| 001        |    70 |
| 002        |    20 |
+------------+-------+
3 rows in set (0.00 sec)

第二种情况,例:

mysql> SELECT * FROM score WHERE score > 65 UNION SELECT * FROM score WHERE id = 4;
+----+------------+------------+-------+
| id | student_id | subject_id | score |
+----+------------+------------+-------+
|  1 | 001        | 1001       |    80 |
|  3 | 001        | 1002       |    70 |
|  4 | 002        | 1002       |  60.5 |
+----+------------+------------+-------+
3 rows in set (0.00 sec)

此种情况下(即情况2),可以用一个SELECT语句加上WHERE条件来实现。

注:

  • UNION中的每个查询必须包含相同的列、表达式或聚集函数。
  • UNION中的每个SELECT语句返回的列类型必须兼容。可以不必完全相同,但是可以隐式转换。
  • UNION默认会消除重复的行,如果要返回所有行:请用UNION ALL。
  • 再用UNION查询时,只能使用一条ORDER BY子句,它必须出现在最后一条SELECT之后。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容

  • layout: posttitle: "MySQL高级查询"date: 2016-06-02 11:14:38 +...
    EwanRenton阅读 348评论 0 2
  • 1.分组查询 1.1什么是分组: 针对于班上所有的同学: 分组情况1-按照性别分组:男生一组,女生一组,之后可以统...
    贾里阅读 1,998评论 1 2
  • 你的眼睛蒙着一层雾。 听人说,敢于对视的人都是诚实的。 可是啊,对视的眼睛如果蒙了一层东西,还是诚实吗? 如果你爱...
    林香文阅读 205评论 0 0
  • 半阳半雨, 一日消度, 坠煮嗒嗒, 乌沉北觑, 瓢秃是靡, 浑油啮啮; 雨歇雷唬, 湿鸟咬咬, 阳火盛盛, 胭脂憔...
    笛木阅读 267评论 0 0
  • 我原本不是一个能闲下心来养花,静下性子喂鱼的人。 各种蹦跶之后,突然间像变了一个人,不爱出动了,有时可以静一整天,...
    琥儿说阅读 462评论 0 0