当前位置:Gxlcms > 数据库问题 > oracle多表查询

oracle多表查询

时间:2021-07-01 10:21:17 帮助过:16人阅读


    笛卡尔积在SQL中的实现方式既是交叉连接(Cross Join)。所有连接方式都会先生成临时笛卡尔积表,笛卡尔积是关系代数里的一个概念,表示两个表中的每一行数据任意组合。

-- 笛卡尔积 
select * from emp, dept;
技术分享


-- 使用公共字段,去掉笛卡尔积
select * from emp, dept where emp.deptno = dept.deptno;
技术分享

-- 查询出每个雇员的姓名,工作,雇员的直接上级领导的姓名
-- 在emp表中的MGR表示一个雇员的上级领导的编号,如果现在要想查询一个雇员的上级领导姓名,则需要用emp表与emp表进行自连接操作
select e.ename, e.job, s.ename from emp s, emp e where s.empno = e.mgr; 

-- 查询出每个雇员的姓名,工作,雇员的直接上级领导的姓名,雇员所在的部门名称
select e.ename, e.job,s.ename mgr_name, d.dname from emp e, emp s, dept d where e.mgr = s.empno and e.deptno = d.deptno;

-- 查询出每个雇员的姓名,工资,部门名称,工资在公司的等级及其领导的姓名,领导的工资,以及领导所对应的等级
SELECT e.ename, e.sal, d.dname,
decode(g.grade,5,'第一等工资',4,'第二等工资',3,'第三等工资',2,'第四等工资',1,'第五等工资')
e_grade, s.ename mgr_name, s.sal mgr_sal,
decode(g2.grade,5,'第一等工资',4,'第二等工资',3,'第三等工资',2,'第四等工资',1,'第五等工资') m_grade
FROM emp e,dept d,salgrade g,emp s,salgrade g2
WHERE (e.deptno=d.deptno) AND (e.sal BETWEEN g.losal AND g.hisal)
AND (s.empno=e.mgr) AND (s.sal BETWEEN g2.losal AND g2.hisal);

左右连接

-- 利用emp 和 dept 做一个连接查询,查询结果包括雇员编号,雇员姓名,部门编号,部门名称,部门所在位置
select e.empno, e.ename, d.deptno, d.dname, d.loc
from emp e,dept d
where e.deptno = d.deptno;
技术分享

-- 通过观察不难发现,在以上结果中部门编号40没有出现,这是因为在进行普通连接时两边都有的才出现在结果中,在emp表中没有部门40,所以结果中也就没有40
select e.empno, e.ename, d.deptno, d.dname, d.loc
from emp e, dept d
where e.deptno(+) = d.deptno;

技术分享

部门40就出现在结果中了.此时我们使用的是右连接。


      (+)在=左边 表示右连接
      (+)在=右边 表示左连接

      左连接以左表为基准,右连接以右表为基准

-- 左连接以左表为基准,右连接以右表为基准
select e.empno, e.ename, d.deptno, d.dname, d.loc
from emp e, dept d
where e.deptno = d.deptno(+);

-- 查询雇员的编号,姓名,及其领导的编号,姓名
select e.empno, e.ename, s.empno, s.ename
from emp e, emp s
where e.mgr = s.empno;

-- 但我们经查询知道emp表中应该有14条记录,在以上结果中缺少了KING,因为KING是最高领导,所以他的领导编号为空,故没办法查询出来.为了显示出来,我们用左/右连接来实现
select e.empno, e.ename, s.empno, s.ename
from emp e, emp s
where e.mgr = s.empno(+);

技术分享


-- 左右连接的连接方式不是固定的,具体采用左连接还是右连接取决于基准条件在等式的左边还是右边,上面的左连接把基准条件放在右边,就变为了右连接
select e.empno, e.ename, s.empno, s.ename
from emp e, emp s
where s.empno(+) = e.mgr;

得到的同样是上图的结果只,这里就不重复贴图了。


SQL:1999SQL定义


语法格式

SELECT table1.column,table2.column

FROM table1 [CROSS JOIN table2]|

[NATURAL JOIN table2]|

[JOIN table2 USING(column_name)]|

[JOIN table2 ON(table1.column_name=table2.column_name)]|

[LEFT|RIGHT|FULL OUTER JOIN table2 ON (table1.column_name=table2.column_name)]


-- 交叉连接(CROSS JOIN):迪卡尔积运算
select * from emp cross join dept;
技术分享


select * from emp cross join dept where emp.empno = 7369;

技术分享

-- 自然连接 NATURAL JOIN
select * from emp natural join dept;

技术分享


-- 根据结果可以发现,自然连接按公共字段相等进行连接,并且产生的结果会自动消除重复的列
select * from emp e, dept d
where e.deptno = d.deptno;

技术分享


-- USING(column_name) 用于指定两个表之间的连接字段
select * from emp join dept using(deptno);

技术分享


-- ON(tab1.column_name=tab2.column_name) 用于指定两表的连接条件
select * from emp join dept on emp.deptno = dept.deptno;

技术分享


-- LEFT[OUTER] JOIN 左连接 OUTER可有可无
select e.empno, e.ename, d.deptno, d.dname, d.loc
from emp e left join dept d
on e.deptno = d.deptno;
select e.empno, e.ename, d.deptno, d.dname, d.loc
from emp e left outer join dept d
on e.deptno = d.deptno;


技术分享


-- RIGHT JOIN 右连接
select e.empno, e.ename, d.deptno, d.dname, d.loc
from emp e right join dept d
on e.deptno = d.deptno;

技术分享


-- INNER JOIN 取交集
create table empbak as select * from emp where emp.empno in (7369, 7499, 7521, 7566, 7654, 7698);
select e1.empno, e1.empno, e1.job, e1.mgr, e1.hiredate, e1.sal, e1.comm, e1.deptno 
from emp e1 inner join empbak e2 on e1.empno = e2.empno;

技术分享


-- FULL OUTER JOIN 取并集
update empbak e set e.empno = 8369 where e.empno = 7369;
update empbak e set e.empno = 8499 where e.empno = 7499;
update empbak e set e.empno = 8521 where e.empno = 7521;
select * from emp e full join empbak s on e.empno = s.empno;

技术分享


-- 对于没有匹配的记录,则会以null做为值
select * from emp e full join empbak s on e.empno = s.empno
where e.empno is not null and s.empno is not null;

技术分享

-- UNION 取并集 去掉重复记录
select * from emp
union
select * from empbak;

技术分享


-- UNION ALL 取并集 不去除重复记录
select * from emp
union all
select * from empbak;

技术分享

最下面的三条记录是重复记录,在上面的union连接中,多余的三条重复记录是被去掉的。




oracle多表查询

标签:select   多表查询   左连接   右连接   

人气教程排行