Oracle学习(6):子查询

子查询

为何要用子查询

SQL> --查询工资比SCOTT高的员工信息

SQL> --1. 知道SCOTT的工资

SQL> select sal from emp where ename=‘SCOTT‘;

SAL

----------

3000

SQL> --2. 查询比3000高的员工

SQL> set linesize 120

SQL> col sal for 9999

SQL> select * from emp where sal>3000;

EMPNO ENAME      JOB              MGR HIREDATE         SAL       COMM     DEPTNO

---------- ---------- --------- ---------- -------------- ----- ---------- ----------

7839 KING       PRESIDENT            17-11月-81      5000                    10

SQL> --子查询解决问题:不能一步求解时,考虑使用子查询

SQL> select *

2  from emp

3  where sal> (select sal

4              from emp

5              where ename=‘SCOTT‘);

EMPNO ENAME      JOB              MGR HIREDATE         SAL       COMM     DEPTNO

---------- ---------- --------- ---------- -------------- ----- ---------- ----------

7839 KING       PRESIDENT            17-11月-81      5000                    10

子查询注意的问题

SQL> /*

SQL> 注意的问题:

SQL> 1. 子查询应该在括号中

SQL> 2. 子查询相对主查询采用缩进

SQL> 3. 可以在主查询的where from select having后,都可以放子查询

SQL> 4. 在主查询的group by后面,不能放置子查询

SQL> 5. 强调在from后面放置子查询

SQL> 6. 主查询和子查询可以不是同一张表,只要子查询返回的结果,主查询可以使用,即可

SQL> 7. 一般来讲,不在子查询中使用 order by;但在Top-N分析中,必须使用order by

SQL> 8. 单行子查询必须使用单行操作符;多行子查询必须使用多行操作符

SQL> 9. 注意:子查询中的空值

SQL> */

子查询 (内查询) 在主查询之前一次执行完成。

子查询的结果被主查询使用 (外查询)。

子查询要包含在括号内。

将子查询放在比较条件的右侧。

单行操作符对应单行子查询,多行操作符对应多行子查询。

子查询放置的位置

SQL> -- 3. 可以在主查询的where from select having后,都可以放子查询

SQL> select *

2  from (select ename,sal from emp);

ENAME        SAL

---------- -----

SMITH        800

ALLEN       1600

WARD        1250

JONES       2975

MARTIN      1250

BLAKE       2850

CLARK       2450

SCOTT       3000

KING        5000

TURNER      1500

ADAMS       1100

ENAME        SAL

---------- -----

JAMES        950

FORD        3000

MILLER      1300

已选择14行。

SQL> ed

已写入 file afiedt.buf

1  select e.*

2* from (select ename,sal from emp) e

SQL> /

ENAME        SAL

---------- -----

SMITH        800

ALLEN       1600

WARD        1250

JONES       2975

MARTIN      1250

BLAKE       2850

CLARK       2450

SCOTT       3000

KING        5000

TURNER      1500

ADAMS       1100

ENAME        SAL

---------- -----

JAMES        950

FORD        3000

MILLER      1300

已选择14行。

SQL> ---select后放子查询:必须是单行子查询

SQL> select ename,sal,(select job from emp where empno=7839)

2  from emp;

ENAME        SAL (SELECTJO

---------- ----- ---------

SMITH        800 PRESIDENT

ALLEN       1600 PRESIDENT

WARD        1250 PRESIDENT

JONES       2975 PRESIDENT

MARTIN      1250 PRESIDENT

BLAKE       2850 PRESIDENT

CLARK       2450 PRESIDENT

SCOTT       3000 PRESIDENT

KING        5000 PRESIDENT

TURNER      1500 PRESIDENT

ADAMS       1100 PRESIDENT

ENAME        SAL (SELECTJO

---------- ----- ---------

JAMES        950 PRESIDENT

FORD        3000 PRESIDENT

MILLER      1300 PRESIDENT

已选择14行。

主查询和子查询是否同表

SQL> --6. 主查询和子查询可以不是同一张表,只要子查询返回的结果,主查询可以使用,即可

SQL> select ename,sal,(select dname from dept where deptno=10)

2  from emp;

ENAME        SAL (SELECTDNAMEFR

---------- ----- --------------

SMITH        800 ACCOUNTING

ALLEN       1600 ACCOUNTING

WARD        1250 ACCOUNTING

JONES       2975 ACCOUNTING

MARTIN      1250 ACCOUNTING

BLAKE       2850 ACCOUNTING

CLARK       2450 ACCOUNTING

SCOTT       3000 ACCOUNTING

KING        5000 ACCOUNTING

TURNER      1500 ACCOUNTING

ADAMS       1100 ACCOUNTING

ENAME        SAL (SELECTDNAMEFR

---------- ----- --------------

JAMES        950 ACCOUNTING

FORD        3000 ACCOUNTING

MILLER      1300 ACCOUNTING

已选择14行。

子查询与多表查询对比

SQL> host cls

SQL> --查询部门名称为SALES的员工信息

SQL> --子查询

1  select *

2  from emp

3  where deptno=(select deptno

4                from dept

5*               where dname=‘SALES‘)

SQL> /

EMPNO ENAME      JOB              MGR HIREDATE         SAL       COMM     DEPTNO

---------- ---------- --------- ---------- -------------- ----- ---------- ----------

7499 ALLEN      SALESMAN        7698 20-2月 -81      1600        300         30

7521 WARD       SALESMAN        7698 22-2月 -81      1250        500         30

7654 MARTIN     SALESMAN        7698 28-9月 -81      1250       1400         30

7698 BLAKE      MANAGER         7839 01-5月 -81      2850                    30

7844 TURNER     SALESMAN        7698 08-9月 -81      1500          0         30

7900 JAMES      CLERK           7698 03-12月-81       950                    30

已选择6行。

SQL> --多表查询

SQL> select e.*

2  from dept d,emp e

3  where d.dname=‘SALES‘ and d.deptno=e.deptno;

EMPNO ENAME      JOB              MGR HIREDATE         SAL       COMM     DEPTNO

---------- ---------- --------- ---------- -------------- ----- ---------- ----------

7499 ALLEN      SALESMAN        7698 20-2月 -81      1600        300         30

7521 WARD       SALESMAN        7698 22-2月 -81      1250        500         30

7654 MARTIN     SALESMAN        7698 28-9月 -81      1250       1400         30

7698 BLAKE      MANAGER         7839 01-5月 -81      2850                    30

7844 TURNER     SALESMAN        7698 08-9月 -81      1500          0         30

7900 JAMES      CLERK           7698 03-12月-81       950                    30

已选择6行。

一个数据库优化问题(子查询与多表查询)

SQL> --如果子查询和多表查询均可,一般采用多表查询         (子查询要查询两次数据库,多表查询查询一次)

单行子查询

子查询结果只返回一行

使用单行比较操作符

查询示例:

SELECTlast_name,job_id,salary

FROM   employees

WHERE 
job_id

                (SELECT
job_id

                 FROM   employees

                 WHERE 
employee_id= 141)

AND    salary >

                (SELECT salary

                 FROM   employees

                 WHERE 
employee_id= 143);

多行子查询

in与not in

SQL> --多行子查询

SQL> --in:  在集合中

SQL> --查询部门名称为SALES和ACCOUNTING的员工信息

SQL> --练习:多表查询

SQL> select *

2  from emp

3  where deptno in   (select deptno from dept where dname=‘SALES‘ or dname=‘ACCOUNTING‘);

EMPNO ENAME      JOB              MGR HIREDATE         SAL       COMM     DEPTNO

---------- ---------- --------- ---------- -------------- ----- ---------- ----------

7499 ALLEN      SALESMAN        7698 20-2月 -81      1600        300         30

7521 WARD       SALESMAN        7698 22-2月 -81      1250        500         30

7654 MARTIN     SALESMAN        7698 28-9月 -81      1250       1400         30

7698 BLAKE      MANAGER         7839 01-5月 -81      2850                    30

7782 CLARK      MANAGER         7839 09-6月 -81      2450                    10

7839 KING       PRESIDENT            17-11月-81      5000                    10

7844 TURNER     SALESMAN        7698 08-9月 -81      1500          0         30

7900 JAMES      CLERK           7698 03-12月-81       950                    30

7934 MILLER     CLERK           7782 23-1月 -82      1300                    10

已选择9行。

SQL> ed

已写入 file afiedt.buf

1  select *

2  from emp

3* where deptno not in (select deptno from dept where dname=‘SALES‘ or dname=‘ACCOUNTING‘)

SQL> /

EMPNO ENAME      JOB              MGR HIREDATE         SAL       COMM     DEPTNO

---------- ---------- --------- ---------- -------------- ----- ---------- ----------

7369 SMITH      CLERK           7902 17-12月-80       800                    20

7566 JONES      MANAGER         7839 02-4月 -81      2975                    20

7788 SCOTT      ANALYST         7566 13-7月 -87      3000                    20

7876 ADAMS      CLERK           7788 13-7月 -87      1100                    20

7902 FORD       ANALYST         7566 03-12月-81      3000                    20

SQL> host cls

any与all查询

any

SQL> --any: 集合中的任意一个

SQL> --查询工资比20号部门任意员工工资高的员工信息

SQL> select *

2  from emp

3  where sal > any (select sal from emp where deptno=20);

EMPNO ENAME      JOB              MGR HIREDATE         SAL       COMM     DEPTNO

---------- ---------- --------- ---------- -------------- ----- ---------- ----------

7839 KING       PRESIDENT            17-11月-81      5000                    10

7902 FORD       ANALYST         7566 03-12月-81      3000                    20

7788 SCOTT      ANALYST         7566 13-7月 -87      3000                    20

7566 JONES      MANAGER         7839 02-4月 -81      2975                    20

7698 BLAKE      MANAGER         7839 01-5月 -81      2850                    30

7782 CLARK      MANAGER         7839 09-6月 -81      2450                    10

7499 ALLEN      SALESMAN        7698 20-2月 -81      1600        300         30

7844 TURNER     SALESMAN        7698 08-9月 -81      1500          0         30

7934 MILLER     CLERK           7782 23-1月 -82      1300                    10

7521 WARD       SALESMAN        7698 22-2月 -81      1250        500         30

7654 MARTIN     SALESMAN        7698 28-9月 -81      1250       1400         30

EMPNO ENAME      JOB              MGR HIREDATE         SAL       COMM     DEPTNO

---------- ---------- --------- ---------- -------------- ----- ---------- ----------

7876 ADAMS      CLERK           7788 13-7月 -87      1100                    20

7900 JAMES      CLERK           7698 03-12月-81       950                    30

已选择13行。

SQL> ed

已写入 file afiedt.buf

1  select *

2  from emp

3* where sal > any (select sal from emp where deptno=10)

SQL> /

EMPNO ENAME      JOB              MGR HIREDATE         SAL       COMM     DEPTNO

---------- ---------- --------- ---------- -------------- ----- ---------- ----------

7839 KING       PRESIDENT            17-11月-81      5000                    10

7902 FORD       ANALYST         7566 03-12月-81      3000                    20

7788 SCOTT      ANALYST         7566 13-7月 -87      3000                    20

7566 JONES      MANAGER         7839 02-4月 -81      2975                    20

7698 BLAKE      MANAGER         7839 01-5月 -81      2850                    30

7782 CLARK      MANAGER         7839 09-6月 -81      2450                    10

7499 ALLEN      SALESMAN        7698 20-2月 -81      1600        300         30

7844 TURNER     SALESMAN        7698 08-9月 -81      1500          0         30

已选择8行。

all

SQL> --all:和集合的所有值比

SQL>  --查询工资比20号部门所有员工工资高的员工信息

SQL> select *

2  from emp\

3  ;

from emp\

*

第 2 行出现错误:

ORA-00911: 无效字符

SQL> select *

2  from emp

3  where sal > all (select sal from emp where deptno=20);

EMPNO ENAME      JOB              MGR HIREDATE         SAL       COMM     DEPTNO

---------- ---------- --------- ---------- -------------- ----- ---------- ----------

7839 KING       PRESIDENT            17-11月-81      5000                    10

SQL> select *

2  from emp

3  where sal > (select min(sal) from emp where deptno=10);

EMPNO ENAME      JOB              MGR HIREDATE         SAL       COMM     DEPTNO

---------- ---------- --------- ---------- -------------- ----- ---------- ----------

7499 ALLEN      SALESMAN        7698 20-2月 -81      1600        300         30

7566 JONES      MANAGER         7839 02-4月 -81      2975                    20

7698 BLAKE      MANAGER         7839 01-5月 -81      2850                    30

7782 CLARK      MANAGER         7839 09-6月 -81      2450                    10

7788 SCOTT      ANALYST         7566 13-7月 -87      3000                    20

7839 KING       PRESIDENT            17-11月-81      5000                    10

7844 TURNER     SALESMAN        7698 08-9月 -81      1500          0         30

7902 FORD       ANALYST         7566 03-12月-81      3000                    20

已选择8行。

SQL> ed

已写入 file afiedt.buf

1  select *

2  from emp

3* where sal > (select max(sal) from emp where deptno=20)

SQL> /

EMPNO ENAME      JOB              MGR HIREDATE         SAL       COMM     DEPTNO

---------- ---------- --------- ---------- -------------- ----- ---------- ----------

7839 KING       PRESIDENT            17-11月-81      5000                    10

SQL> host cls

SQL> --查询不是经理的员工

SQL> select * from emp;

EMPNO ENAME      JOB              MGR HIREDATE         SAL       COMM     DEPTNO

---------- ---------- --------- ---------- -------------- ----- ---------- ----------

7369 SMITH      CLERK           7902 17-12月-80       800                    20

7499 ALLEN      SALESMAN        7698 20-2月 -81      1600        300         30

7521 WARD       SALESMAN        7698 22-2月 -81      1250        500         30

7566 JONES      MANAGER         7839 02-4月 -81      2975                    20

7654 MARTIN     SALESMAN        7698 28-9月 -81      1250       1400         30

7698 BLAKE      MANAGER         7839 01-5月 -81      2850                    30

7782 CLARK      MANAGER         7839 09-6月 -81      2450                    10

7788 SCOTT      ANALYST         7566 13-7月 -87      3000                    20

7839 KING       PRESIDENT            17-11月-81      5000                    10

7844 TURNER     SALESMAN        7698 08-9月 -81      1500          0         30

7876 ADAMS      CLERK           7788 13-7月 -87      1100                    20

EMPNO ENAME      JOB              MGR HIREDATE         SAL       COMM     DEPTNO

---------- ---------- --------- ---------- -------------- ----- ---------- ----------

7900 JAMES      CLERK           7698 03-12月-81       950                    30

7902 FORD       ANALYST         7566 03-12月-81      3000                    20

7934 MILLER     CLERK           7782 23-1月 -82      1300                    10

已选择14行。

关于not in与in后面的空值问题

not in相当于all的范围,即非集合中的所有,所以集合中出现null时,返回空,即结果集为空集!

in相当于any的范围,即在集合中的任意一个元素符合即可,所以即使集合中出现null,也不妨碍结果集!

SQL> select *

2  from emp

3  where empno not in (select mgr from emp);

未选定行

SQL>  --查询是经理的员工

SQL> ed

已写入 file afiedt.buf

1  select *

2  from emp

3* where empno in (select mgr from emp)

SQL> /

EMPNO ENAME      JOB              MGR HIREDATE         SAL       COMM     DEPTNO

---------- ---------- --------- ---------- -------------- ----- ---------- ----------

7902 FORD       ANALYST         7566 03-12月-81      3000                    20

7698 BLAKE      MANAGER         7839 01-5月 -81      2850                    30

7839 KING       PRESIDENT            17-11月-81      5000                    10

7566 JONES      MANAGER         7839 02-4月 -81      2975                    20

7788 SCOTT      ANALYST         7566 13-7月 -87      3000                    20

7782 CLARK      MANAGER         7839 09-6月 -81      2450                    10

已选择6行。

SQL> select *

2  from emp

3  where empno not in (select mgr from emp where mgr is not null);

EMPNO ENAME      JOB              MGR HIREDATE         SAL       COMM     DEPTNO

---------- ---------- --------- ---------- -------------- ----- ---------- ----------

7844 TURNER     SALESMAN        7698 08-9月 -81      1500          0         30

7521 WARD       SALESMAN        7698 22-2月 -81      1250        500         30

7654 MARTIN     SALESMAN        7698 28-9月 -81      1250       1400         30

7499 ALLEN      SALESMAN        7698 20-2月 -81      1600        300         30

7934 MILLER     CLERK           7782 23-1月 -82      1300                    10

7369 SMITH      CLERK           7902 17-12月-80       800                    20

7876 ADAMS      CLERK           7788 13-7月 -87      1100                    20

7900 JAMES      CLERK           7698 03-12月-81       950                    30

已选择8行。

SQL> spool off

时间: 2024-12-13 13:08:29

Oracle学习(6):子查询的相关文章

彻底搞懂oracle的标量子查询

oracle标量子查询和自定义函数有时用起来比较方便,而且开发人员也经常使用,数据量小还无所谓,数据量大,往往存在性能问题. 以下测试帮助大家彻底搞懂标量子查询. SQL> create table a (id int,name varchar2(10)); Table created. SQL> create table b (id int,name varchar2(10)); Table created. SQL> insert into a values (1,'a1'); 1

[转]Oracle DB 使用子查询来解决查询

? 定义子查询 ? 描述子查询可以解决的问题类型 ? 列出子查询的类型 ? 编写单行和多行子查询 ? 子查询:类型.语法和准则 ? 单行子查询: – 子查询中的组函数 – 带有子查询的HAVING子句 ? 多行子查询 – 使用ALL或ANY运算符 ? 使用EXISTS运算符 ? 子查询中的空值 使用子查询解决问题 谁的薪金高于Abel 的薪金? 子查询: Abel 的薪金是多少? 主查询: 哪些雇员的薪金高于Abel 的薪金? 使用子查询解决问题 假设要编写一个查询来找出谁的薪金高于Abel 的

MySQL学习笔记-子查询和连接

MySQL学习笔记-子查询和连接 使客户端进入gbk编码方式显示: mysql> SET NAMES gbk; 1.子查询 子查询的定义: 子查询(Subquery)是指出现在其他SQL语句内的SELECT子句. 例如:  SELECT * FROM t1 WHERE col1 = (SELECT col2 FROM t2); 其中SELECT * FROM t1 称为Outer Query / Outer Statement (外部查询) SELECT col2 FROM t2 , 被称为Su

Oracle with重用子查询

--with 重用子查询对于多次使用相同子查询的复杂查询语句来说,用户可能会将查询语句分成两条语句执行.第一条语句将子查询结果存放到临时表,第二条查询语句使用临时表处理数据.从 Oracle 9i 开始,通过 with 子句可以给予子查询指定一个名称,并且使得在一条语句中可以完成所有任务,从而避免了使用临时表. [email protected] test10g> with summary as (  2  select dname, sum(sal) dept_total from emp,

Oracle DB 使用子查询来解决查询

? 定义子查询 ? 描述子查询可以解决的问题类型 ? 列出子查询的类型 ? 编写单行和多行子查询 ? 子查询:类型.语法和准则 ? 单行子查询: – 子查询中的组函数 – 带有子查询的HAVING子句 ? 多行子查询 – 使用ALL或ANY运算符 ? 使用EXISTS运算符 ? 子查询中的空值 使用子查询解决问题 谁的薪金高于Abel 的薪金? 子查询: Abel 的薪金是多少? 主查询: 哪些雇员的薪金高于Abel 的薪金? 使用子查询解决问题 假设要编写一个查询来找出谁的薪金高于Abel 的

MySql学习(三) —— 子查询(where、from、exists) 及 连接查询(left join、right join、inner join、union join)

同样的,使用goods表来练习子查询,表结构如下: 所有数据(cat_id与category.cat_id关联): 类别表: mingoods(连接查询时作测试) 一.子查询 1.where型子查询:把内层查询的结果作为外层查询的比较条件 1.1 查询id最大的一件商品(使用排序+分页实现) :mysql> SELECT goods_id,goods_name,shop_price FROM goods ORDER BY goods_id DESC LIMIT 1; 1.2 查询id最大的一件商

Oracle 笔记之子查询

子查询 当我们的一个操作需要基于另一个查询记过,那么就先行执行的这个查询就是子查询 子查询分为: 单行单列子查询:查的结果只有一行,且只有一个字段 多行单列子查询:查询出来的结果有多行,但只有一列 多行多列子查询 查询出多行多个列. 通常,单行单列与多行多列子查询用于where子句中而多行多列子查询用于 FROM子句中. --查看和SCOTT相同职位的其他员工 SELECT ename,sal,job FROM emp WHERE job=(SELECT job FROM emp WHERE e

Oracle篇 之 子查询

子查询:先执行内部再外部 Select last_name,salary,dept_id From s_emp Where dept_id in ( Select dept_id From s_emp Where dept_id is not null ); Select last_name,dept_id From s_emp Where dept_id = ( Select dept_id From s_emp Where last_name=’Smith’ ); Select salary

Oracle系列七 子查询

子查询语法 SELECT select_list FROM table WHERE expr operator (SELECT select_list FROM table); 子查询 (内查询) 在主查询之前一次执行完成. 子查询的结果被主查询(外查询)使用 . 示例: SELECT last_name FROM employees WHERE salary > ( SELECT salary FROM employees WHERE last_name = 'Abel' ); 注意事项 子查

Oracle 数据库基础学习 (六) 子查询

1.内连接(等值连接) 示例:将连个表内容连接显示 select * from dept d, emp e where d.deptno = e.deptno; 注意:只有当d.deptno = e.deptno条件满足,内容才会显示,否则不显示 2 .外连接:让等值判断左右两边有一边的数据可以全部显示出来,使用外连接方式时使用”(+)” 格式:   左外连接:字段=字段(+) 右外连接:字段(+)=字段 示例:显示雇员的姓名,职位和领导姓名 分析: 确认要使用的数据: 1.  emp表的雇员姓