详解Mariadb SELECT子查询及UNION

SELECT子查询

嵌套在其他SELECT语句中的SELECT查询叫做子查询,为什么要这样做呢?其实我们已经学了多表查询,很多时候多表查询已经够用了?但是子查询又有自身存在的地位和价值,还拿多表查询那个例子来说。

我们想知道某一城市所使用的语言,就可以分为两个步骤:

1.在City表中查询该城市的CountryCode。

2.使用查询到的这个CountryCode在CountryLanguage表中查询该国家所使用的语言。

虽然,可以分两步完成,但是,需要两次查询和两次传输,在带宽和性能的对比下,我们更希望让Mysql(MariaDB)来帮助我们完成这件事不是吗?

看下用子查询是什么样的~

MariaDB [world]> SELECT Language FROM countrylanguage WHERE CountryCode = (SELECT CountryCode FROM city WHERE Name = 'Peking');
+-----------+
| Language  |
+-----------+
| Chinese   |
| Dong      |
| Hui       |
| Mantšu    |
| Miao      |
| Mongolian |
| Puyi      |
| Tibetan   |
| Tujia     |
| Uighur    |
| Yi        |
| Zhuang    |
+-----------+
12 rows in set (0.07 sec)

看到(SELECT CountryCode FROM city WHERE Name = ‘Peking’)这一坨了吗?用括号括起来的这个查询,他会得到北京的CountryCode,而这个CountryCode又作为外面SELECT的WHERE检索条件。

所以,子查询就是用括号括起来的查询,而MariaDB会在进行查询时先进行括号内的查询得到一个值或一组值替换到相应的位置

ANY或SOME子查询

其实ANY和SOME跟IN的意思是一样的,只要满足操作符对()内的任一值的操作为TRUE即可,如下所示。

查询所有技术部(Tech)和销售部(Sales)的员工:

MariaDB [world]> SELECT * FROM user                                                    
   -> WHERE deptid IN
   -> (SELECT id FROM department WHERE name IN ('Sales','Tech'));

MariaDB [world]> SELECT * FROM user
   -> WHERE deptid = ANY
   -> (SELECT id FROM department WHERE name IN ('Sales','Tech'));

MariaDB [world]> SELECT * FROM user
   -> WHERE deptid = SOME
   -> (SELECT id FROM department WHERE name IN ('Sales','Tech'));


+----+-------+----------+---------------------+--------+
| id | name  | password | regtime             | deptid |
+----+-------+----------+---------------------+--------+
|  1 | test  | test     | 2018-03-05 17:25:26 |      1 |
|  2 | test1 | test1    | 2018-03-05 17:25:26 |      1 |
|  3 | lucy  | lucy     | 2018-03-05 17:25:26 |      2 |
+----+-------+----------+---------------------+--------+
3 rows in set (0.00 sec)

EXISTS存在判断

只要子查询返回的有值即为TRUE,否则即为FALSE,如下例展示:

MariaDB [world]> SELECT EXISTS( SELECT * FROM user WHERE deptid = (SELECT id FROM department WHERE name='Tech') ) AS dep_is_Exist;
+--------------+
| dep_is_Exist |
+--------------+
|            1 |
+--------------+
1 row in set (0.01 sec)

当然EXISTS前可以加一个NOT,这样就变成了当子查询没有结果时为真了。

注意事项

  1. 作为子查询的SELECT语句只能查询单个列,企图检索多个列将返回错误。
  2. 通常子查询和表连接可以做相互转换,而表连接相对而言会比子查询获得更好的效率。

UNION组合表

SQL允许执行多个查询(多条SELECT语句),并将结果作为一个查询结果集返回,这些组合查询通常称为并(union)或复合查询(compound query)。

需要注意如下几点:

  1. UNION必须由两条或两条以上的SELECT语句组成,语句之间用关键字UNION分隔(因此,如果组合四条SELECT语句,将要使用三个UNION关键字)。
  2. UNION中的每个查询必须包含相同的列、表达式或聚集函数(不过,各个列不需要以相同的次序列出)。
  3. 列数据类型必须兼容:类型不必完全相同,但必须是DBMS可以隐含转换的类型(例如,不同的数值类型或不同的日期类型)。

语法如下:

SELECT column_name FROM table1
UNION
SELECT column_name FROM table2

看如下例子,检索所有美国和中国的国家:

MariaDB [world]> SELECT * FROM city WHERE CountryCode = 'CHN'
   -> UNION
   -> SELECT * FROM city WHERE CountryCode ='USA';
+------+-------------------------+-------------+----------------------+------------+
| ID   | Name                    | CountryCode | District             | Population |
+------+-------------------------+-------------+----------------------+------------+
| 1890 | Shanghai                | CHN         | Shanghai             |    9696300 |
| 1891 | Peking                  | CHN         | Peking               |    7472000 |
....................................................................................
| 4064 | Odessa                  | USA         | Texas                |      89293 |
| 4065 | Carson                  | USA         | California           |      89089 |
| 4066 | Charleston              | USA         | South Carolina       |      89063 |
+------+-------------------------+-------------+----------------------+------------+
637 rows in set (0.01 sec)

//以下这条语句等同上方的SELECT查询
SELECT * FROM city WHERE CountryCode ='USA' OR CountryCode = 'CHN';

统计美国和中国的城市数:

MariaDB [world]> SELECT CountryCode,COUNT(id) AS city_nums FROM city WHERE CountryCode = 'CHN'
   -> UNION
   -> SELECT COUNT(id),CountryCode FROM city WHERE CountryCode = 'USA';
+-------------+-----------+
| CountryCode | city_nums |
+-------------+-----------+
| CHN         | 363       |
| 274         | USA       |
+-------------+-----------+
2 rows in set (0.00 sec)

所以只要列数相同,都是可以组合成同一个结果集的,以下例子第一行显示了美国的城市数,第二行显示了美国的语言数量:

MariaDB [world]> SELECT COUNT(id),CountryCode FROM city WHERE CountryCode = 'USA' UNION SELECT COUNT(Language),CountryCode FROM countrylanguage WHERE CountryCode ='USA';
+-----------+-------------+
| COUNT(id) | CountryCode |
+-----------+-------------+
|       274 | USA         |
|        12 | USA         |
+-----------+-------------+
2 rows in set (0.00 sec)

原创文章,作者:晴川运维,如若转载,请注明出处:https://baike.qcidc.com/16028.html

(0)
晴川运维晴川运维
上一篇 2025年10月21日
下一篇 2025年10月21日

相关推荐

  • 详解docker数据管理

    在docker的使用过程中,势必需要查看容器内应用产生的数据,或者需要将容器内数据进行备份,甚至多个容器之间进行数据共享,这必然会涉及到容器的数据管理: 数据的管理目前提供如下两种…

    Linux系统 2025年10月10日
  • 详解grep命令的排除功能

    grep命令是linux中一种强大的文本搜索工具,它能使用正则表达式搜索文本,并把匹配的行打印出来。 在一个或多个文件中搜素字符串模式,如果字符串模式包括空格,也必须被引用,模式后…

    Linux系统 2025年6月8日
  • 快速上手Zabbix使用方法

    Zabbix是一款能够监控各种网络参数以及服务器健康性和完整性的软件。Zabbix使用灵活的通知机制,允许用户为几乎任何事件配置基于邮件的告警。这样可以快速反馈服务器的问题。基于已…

    Linux系统 2025年6月18日
  • Linux 中安装负载工具 ttyload

    ttyload 是一个轻量级的实用程序,它为 Linux 和其他类 Unix 系统上提供随着时间变化的彩色平均负载。它实现了在终端中(“tty”)图形化跟踪系统的平均负载。 它已知…

    Linux系统 2025年10月7日
  • 详解Kubernetes中的网络类型

    随着Kubernetes王者时代的到来,计算、网络、存储、安全是Kubernetes绕不开的话题,本次主要分享Kubernetes中的网络类型,,后续还会有Kubernetes其它…

    Linux系统 2025年10月20日
  • 我最喜欢用的 5 个 Ansible 模块

    了解如何通过这些 Ansible 模块实现几乎任何事情。 在我成长的时候,我爷爷在他的花园里有一个棚子。他经常会花几个小时在那里制作和修复东西。这是在我们有互联网之前的事情,所以我…

    Linux系统 2025年9月23日
  • Java中JAR包、EAR包、WAR包有什么区别?

    Java中的JAR包、EAR包、WAR包你知道有什么区别吗?本篇文章为大家讲解一下Java中JAR包、EAR包、WAR包。 WAR包 WAR(Web Archive file)网络…

    Linux系统 2025年7月10日
  • echo命令使用实例

    echo命令是linux中最基础的命令,也是很常用的命令,特别是在写shell脚本的时候,可能会经常被用到,虽然echo命令非常基础,但是功能还算丰富,本篇文章为大家分享一下ech…

    Linux系统 2025年6月8日
  • 如何使用 Ansible 安装软件

    使用 Ansible 剧本自动安装和更新设备上的软件。 Ansible 是系统管理员和开发人员用来保持计算机系统处于最佳状态的一种流行的自动化工具。与可扩展框架一样,Ansible…

    Linux系统 2025年6月8日
  • 讲解一下Varnish安装与配置

    Varnish作用是访问web速度的web加速器,被安装在web服务器之前,从而缓存web服务器的应用程序和数据,最后相应客户的请求。 功能与Squid服务器相似,都可以用来做HT…

    Linux系统 2025年9月21日

发表回复

登录后才能评论