Sql子查询

   Sql子查询即是当一个查询语句嵌套在另一个查询语句的查询条件中时叫做子查询也叫做嵌套子查询。根据子查询的返回结果不同,子查询又可分为单行子查询,多行子查询与多列子查询。现在如下三类子查询作说明,并假设有如下四个表记录学生选修课的成绩: Student(stId, stName); course(coId,coName); teacher(teId, teName); score(stId, teId, coId, score) 其中学生可以有多门选修课,一个老师也可以教多门课程。
   1> 单行子查询
   单行子查询是指子查询的返回结果只有一行数据。当主查询语句的条件语句中引用子查询结果时可用单行比较符号(=, >, <, >=, <=, <>)来进行比较。
   例子:查询名叫Kaka的学生的所有成绩 sql> Select score from score where stId=(select stId from Student where stName='Kaka');
   2> 多行子查询
   多行子查询即是子查询的返回结果是多行数据。当主查询语句的条件语句中引用子查询结果时必须用多行比较符号(IN,ALL,ANY)来进行比较。其中,IN的含义是匹配子查询结果中的任一个值即可,ALL则必须要符合子查询的所有值才可,ANY要符合子查询结果的任何一个值即可。而且须注意ALL 和ANY 操作符不能单独使用,而只能与单行比较符(=、>、< 、>= 、<= 、<>)结合使用。
   多行子查询使用IN操作符号例子:查询选修了老师名叫Rona(假设唯一)的学生名字 sql> select stName from Student where stId in(selectdistinct stId from score where teId=(select teId from teacher where teName='Rona'));
   多行子查询使用ALL操作符号例子:查询有一门以上的成绩高于Kaka的最高成绩的学生的名字 sql> select stName from Student where stId in(select distinct stId from score where score >all(select score from score where stId=(select stId from Student where stName= 'Kaka') ));
   多行子查询使用ANY操作符号例子:查询有一门以上的成绩高于Kaka的任何一门成绩的学生的名字 sql> select stName from Student where stId in(select distinct stId from score where score >any(select score from score where stId=(select stId from Student where stName='Kaka')));
   3> 多列子查询
   以上的单行与多行子查询的返回结果都是单列数据,而这个多例子查询顾名义当然就是返回结果是多列的数据。而多列子查询当然也可以返回单行或是多行的多列数据,当是单行多列的子查询时,主查询语句的条件语句中引用子查询结果时可用单行比较符号(=, >, <, >=, <=, <>)来进行比较;当是多行多列子查询时,主查询语句的条件语句中引用子查询结果时必须用多行比较符号(IN,ALL,ANY)来进行比较。
   具体跟以上两个并无太多不同,只是比较的不是单列而是多列数据而已。

### SQL 子查询使用指南与示例 SQL 子查询是指在一个 SQL 查询语句中嵌套另一个查询语句。子查询可以出现在 `SELECT`、`INSERT`、`UPDATE` 或 `DELETE` 语句中,通常用于提供动态数据或条件过滤。以下是一些常见的子查询用法及其示例。 #### 1. 基本子查询 基本子查询通常用于从表中检索单一值,并将其作为主查询的条件。 ```sql SELECT ename, sal FROM emp WHERE sal > (SELECT AVG(sal) FROM emp); ``` 上述查询返回所有薪资高于平均薪资的员工姓名和薪资[^1]。 #### 2. 相关子查询 相关子查询是指子查询依赖于外部查询中的列,每次执行外部查询时,子查询都会重新计算。 ```sql SELECT e.* FROM employees e WHERE salary > (SELECT AVG(salary) FROM employees WHERE dept_id = e.dept_id); ``` 此查询返回每个部门中薪资高于该部门平均薪资的员工信息[^2]。 #### 3. 内联视图子查询 内联视图子查询是将子查询结果集作为虚拟表使用,通常用于排序或限制结果行数。 ```sql -- 获取按薪资排序的所有员工 SELECT ename, job, sal, rownum FROM (SELECT ename, job, sal FROM emp ORDER BY sal); -- 获取薪资最低的前5名员工 SELECT ename, job, sal, rownum FROM (SELECT ename, job, sal FROM emp ORDER BY sal) WHERE rownum <= 5; ``` 这些查询展示了如何通过子查询结果进行排序和限制[^4]。 #### 4. 在 `HAVING` 子句中使用子查询 在聚合函数中使用子查询可以帮助实现更复杂的过滤条件。 ```sql SELECT deptno, job, AVG(sal) FROM emp GROUP BY deptno, job HAVING AVG(sal) > (SELECT sal FROM emp WHERE ename = 'MARTIN'); ``` 此查询返回所有平均薪资高于员工 MARTIN 薪资的部门及其职位。 #### 5. 子查询优化技巧 对于相关子查询,由于每次外部查询都会重新计算子查询,因此性能可能较差。可以通过索引优化或改写为连接查询来提高效率[^3]。 ```sql -- 使用连接查询替代相关子查询 SELECT e1.* FROM employees e1 JOIN (SELECT dept_id, AVG(salary) AS avg_salary FROM employees GROUP BY dept_id) e2 ON e1.dept_id = e2.dept_id WHERE e1.salary > e2.avg_salary; ``` ### 注意事项 - 子查询必须包含在圆括号中。 - 子查询可以返回单个值、一行或多行,但需要根据上下文正确使用。 - 对于复杂查询,建议使用索引或优化查询逻辑以提高性能。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值