在SQL语句中,子查询需要放在圆括号中,在执行子查询时,其执行的过程是先查询出来子查询的结果,然后将子查询返回的结果作为其外层查询的查询条件。子查询根据返回的结果,可以分为单行子查询,多行子查询和多列子查询;根据返回的数据与外层查询之间的关系,可以分为相关子查询和不相关子查询。应该说,子查询在实际开发过程中有着非常广泛的应用。
单行子查询
在实际应用中,如果开发人员或者是用户明确知道其SQL语句中使用的子查询返回的结果是一行数据时,即子查询中返回的结果是一个值时,就可以使用算术比较运算符进行子查询的操作。其中,比较运算符包括=(等于)、>=(大于等于)、<=(小于等于)、>(大于)、<(小于)、!=(不等于)、 <>(不等于)、!>(不大于)、!<(不小于)。
多行子查询
所谓多行子查询是指子查询中返回的结果集中含有多行数据。当子查询返回的是多行数据时,需要使用多行运算符。多行运算符包括IN、ANY、ALL等运算符。
使用IN运算符,可以将满足列表中满足指定表达式的任何一个值都查询出来。在子查询中使用IN 运算符,则与子查询中查询出来的结果集中的任何一个值匹配的结果都会被查询出来。
ANY运算符也用于多行子查询中。ANY运算符的含义是只要与子查询中的任何一个结果值匹配,其值都会被返回。ANY运算符在使用时需要和比较运算符(=(等于)、>=(大于等于)、<=(小于等于)、>(大于)、<(小于)、!=(不等于)、 <>(不等于))放在一起使用。SOME是ANY的一个同义词。
ALL运算符也用于多行子查询中。ALL运算符的含义是与子查询中的所有的结果值匹配时,其值才会被返回。ALL运算符在使用时需要和比较运算符(=(等于)、>=(大于等于)、<=(小于等于)、>(大于)、<(小于)、!=(不等于)、 <>(不等于))放在一起使用。
多列子查询
所谓多列子查询,是指子查询的语句会返回多个数据列的子查询语句。在WHERE子句中也可以使用将多个属性值用括号括起来的方式实现多列子查询。在多列子查询中,WHERE子句中需要使用括号将多个属性括在一起,多个属性之间需要用逗号分开。
外层查询的WHERE子句中根据多列子查询返回的行数不同,可以选择使用不同的运算符。如果多列子查询中返回的数据行是单行的,即返回的结果值只有一个,则可以使用算术比较运算符;如果子查询中返回的数据行是多行的,即返回的结果值不只一个,则可以使用IN、ANY、ALL运算符。
SELECT teaID,teaName,age,sex,dept,profession FROM T_teacher WHERE (dept,profession) = (SELECT dept,profession FROM T_teacher WHERE teaID = 't103265')
相关子查询
在前面介绍的SQL语句子查询中,都是首先执行内层子查询的语句,然后将子查询返回的结果作为外层查询的查询条件检索数据的。这时的子查询只执行一次。与主查询相关的子查询需要引用主查询里的值,所以必须依赖于主查询,因为这种联系,与主查询相关的子查询不能脱离主查询作为一条独立的查询命令执行。
SELECT j FROM t2 where j IN (SELECT i FROM t1); -- 与主查询无关的子查询 SELECT j FROM t2 where (SELECT i FROM t1 WHERE i = j); -- 与主查询相关的子查询
而相关子查询中,子查询需要重复执行。每处理一行外部的查询语句,子查询都会被执行一次。也就是说,相关子查询需要依赖与外层查询,外层查询和子查询之间是存在联系的。与主查询相关的子查询的工作情况是:把值从主查询传递到子查询,看她们是否满足在自查询中给出的条件。通常用EXISTS关键字或者NOT EXISTS关键字实现相关子查询。带有EXISTS关键字的子查询在执行时只会返回逻辑值TRUE或者TALSE,而不会返回任何数据。也就是说,带有EXISTS关键字的子查询不关心返回的是什么数据,而只关心返回的数据“有还是没有”。(测试某个子查询是否返回了数据行)。
使用NOT EXISTS关键字实行相关子查询的查询方法和使用 EXISTS关键字实行相关子查询的查询方法正好相反。如果NOT EXISTS子句的子查询中没有返回的结果,则外层查询的WHERE子句就返回TRUE,则此最终查询的结果集显示出来;如果NOT EXISTS子句的子查询中有返回的结果,则外层查询的WHERE子句就返回FALSE。
在SQL语句中使用子查询
子查询语句除了可以应用在WHERE子句中,也可以应用在SELECT子句、FROM子句、ORDER BY子句、HAVING子句、CREATE TABLE 语句、CREATE VIEW 语句、INSERT 语句、UPDATE语句、 DELETE等语句中。
在SELECT子句中使用子查询,查询学生编号为s102203的成绩信息
SELECT R.stuID, (SELECT stuName FROM T_student WHERE stuID = R.stuID) AS stuName, R.result,R.curID FROM T_result R WHERE R.stuID = 's102203' ORDER BY R.result ASC
在FROM子句中使用子查询,该子查询查询出来的结果集组成一个临时的数据表。查询学生编号为s102203学生的选课成绩信息
SELECT R.stuID,C.curID, C.curName,R.result FROM T_curriculum C, (SELECT curID,stuID,result FROM T_result)R WHERE R.curID=C.curID AND R.stuID = 's102203' ORDER BY R.result ASC
在HAVING子句中可以使用子查询,该子查询查询出来的结果集组成一个临时的数据表。查询以学生编号s2开头的学生的平均成绩。
SELECT R.stuID, AVG(R.result) FROM T_result R,T_curriculum C WHERE R.curID=C.curID GROUP BY R.stuID HAVING R.stuID IN (SELECT stuID FROM T_student WHERE stuID LIKE 's2%') ORDER BY R.stuID
多重子查询允许查询条件中有多个子查询语句。查询教师信息表中职称与教师编号为t181585教师相同但工资比该教师高的教师信息。
SELECT teaID,teaName,age,sex,dept,profession,salary FROM T_teacher WHERE profession = (SELECT profession FROM T_teacher WHERE teaID = 't181585') AND salary> (SELECT salary FROM T_teacher WHERE teaID = 't181585')
在CREATE TABLE语句中使用子查询实现数据表的复制,通过在CREATE TABLE语句中使用子查询可以在建立一张新的数据表的同时将原有表中的数据插入到新建的数据表中,即实现数据表中数据的复制功能。语法格式如下:
CREATE TABLE 表名 AS SELECT语句