sql的join分為三種,內連接、外連接、交叉連接。
以下先建2張表,插入一些數據,后續理解起來更方便一些。
create table emp(empno int, name char(20),depart int);
create table depart(dpno int,dpname char(20));
insert into emp values (1,'bell',1);
insert into emp values (2,'smith',2);
insert into emp values (3,'jet',3);
insert into depart values (1,'design');
insert into depart values (2,'database');
insert into depart values (4,'warehouse');
1. 內連接 inner join 僅列出兩表能按照join條件連接起來的信息,其他的信息不顯示
select a.*,b.* from emp a inner join depart b on a.depart=b.dpno;
empno name depart dpno dpname
----------------------------------------------------------------------
1 bell 1 1 design
2 smith 2 2 database
和如下語句得到的信息是一樣的:
select a.*,b.* from emp a,depart b where a.depart=b.dpno;
內連接:不以哪個表為基礎,僅取出匹配的數據
2. 外連接 outer join
(1)左外連接 left outer join = left join 顯示左表的所有記錄,右表符合join條件的信息顯示,不符合的置空。
select a.*,b.* from emp a left join depart b on a.depart=b.dpno;
empno name depart dpno dpname
----------------------------------------------------------------------
1 bell 1 1 design
2 smith 2 2 database
3 jet 3
(2)右外連接 right outer join = right join 和left join相反,顯示右表的所有記錄,左表符合join條件的信息顯示,不符合的置空。
select a.*,b.* from emp a right join depart b on a.depart=b.dpno;
empno name depart dpno dpname
----------------------------------------------------------------------
1 bell 1 1 design
2 smith 2 2 database
4 warehouse
右連接:以b表為基礎,首先取出b表中所有數據,然后再加上與a,b匹配的的數據
(3)全外連接 full outer join = full join ,左右兩表的信息都全部顯示,符合join條件的信息顯示,不符合的置空。
select a.*,b.* from emp a full join depart b on a.depart=b.dpno;
empno name depart dpno dpname
----------------------------------------------------------------------
1 bell 1 1 design
2 smith 2 2 database
4 warehouse
3 jet 3
3.交叉連接,也即生成兩表的笛卡爾積。得到的記錄相當於兩表記錄的乘積。
select a.*,b.* from emp a cross join depart b ;
empno name depart dpno dpname
----------------------------------------------------------------------
1 bell 1 1 design
2 smith 2 1 design
3 jet 3 1 design
1 bell 1 2 database
2 smith 2 2 database
3 jet 3 2 database
1 bell 1 4 warehouse
2 smith 2 4 warehouse
3 jet 3 4 warehouse
相當於如下語句:
select a.*,b.* from emp a,depart b;
需要注意的是交叉連接后只能跟where,不可以跟on。
如下語句是錯的:
select a.*,b.* from emp a cross join depart b on a.depart=b.dpno;
要修改為這樣的才正確:
select a.*,b.* from emp a cross join depart b where a.depart=b.dpno;
一般來講,在大表關聯的時候,建議使用inner join或者left join,不建議使用cross join或者where
比如:
select a.*,b.* from emp a,depart b where a.depart=b.dpno ; A ---不推薦
select a.*,b.* from emp a cross join depart b where a.depart=b.dpno ; B ---不推薦
select a.*,b.* from emp a inner join depart b on a.depart = b.dpno; C ----推薦
這是因為A,B中,sql需要先對兩表生成笛卡爾積,然后才根據where后的條件進行判斷,而使用C則不需要。所以C較有效率。
oracle Join操作語法錯誤和Join語句的嵌套
oracle自9i之后開始支持 Join 操作,而且支持嵌套,但必須使用括弧,否則會報 from 子句錯誤。
若要創建一個只包括在被聯接字段中具有相同數據的記錄,使用 INNER JOIN 操作。
一、LEFT JOIN 或 RIGHT JOIN 可以嵌套到 INNER JOIN 語句中,
INNER JOIN 語句不能嵌套到 LEFT JOIN 或 RIGHT JOIN 語句中。
二、[/b]對 Join 后的數據集[/b]不能賦別名,賦別名后提示 Join 操作語法錯誤。
三、LEFT JOIN 或 LEFT OUTER JOIN。
左向外聯接的結果集包括 LEFT OUTER 子句中指定的左表的所有行,而不僅僅是聯接列所匹配的行。如果左表的某行在右表中沒有匹配行,則在相關聯的結果集行中右表的所有選擇列表列均為空值。
一般情況下可以等同看待。
建表a
create table a as select rownum as id,rownum as name from dual connect by level < 10;
建表b
create table b as select rownum+5 as id,rownum+5 as name from dual connect by level < 10;
1、左連接
SQL> select * from a left join b on a.id=b.id order by a.id;
ID NAME ID NAME
---------- ---------- ---------- ----------
1 1
2 2
3 3
4 4
5 5
6 6 6 6
7 7 7 7
8 8 8 8
9 9 9 9
左連接:以a表為基礎,首先取出a表中所有數據,然后再加上與a,b匹配的的數據
這樣寫更清晰:
SQL> select a.id aid,a.name aname,b.id bid,b.name bname from a left join b
2 on a.id=b.id order by a.id;
AID ANAME BID BNAME
---------- ---------- ---------- ----------
1 1
2 2
3 3
4 4
5 5
6 6 6 6
7 7 7 7
8 8 8 8
9 9 9 9
這樣寫錯誤:
SQL> select a.id aid,a.name aname,b.id bid,b.name bname from a left join b
2 on aid=bid order by aid;
on aid=bid order by aid
*
ERROR at line 2:
ORA-00904: "BID": invalid identifier
因為上面‘二’說過:對 Join 后的數據集不能賦別名,賦別名后提示 Join 操作語法錯誤。
這樣寫也錯誤:
SQL> select a.id aid,a.name aname,b.id bid,b.name bname from a
2 left join b
3 order by a.id
4 where a.id=b.id;
order by a.id
*
ERROR at line 3:
ORA-00905: missing keyword
order by和group by都不能出現在where前面。
oracle自9i之后開始支持 Join 操作,而且支持嵌套,但必須使用括弧,否則會報 from 子句錯誤。 三、LEFT JOIN 或 LEFT OUTER JOIN。 一般情況下可以等同看待。 建表a SQL> select * from a; ID NAME SQL> select * from b; ID NAME 兩個表a,b相連接,要取出id相同的字段,如下:
1、左連接 SQL> select * from a left join b on a.id=b.id order by a.id; ID NAME ID NAME 左連接:以a表為基礎,首先取出a表中所有數據,然后再加上與a,b匹配的的數據 這樣寫更清晰: SQL> select a.id aid,a.name aname,b.id bid,b.name bname from a left join b AID ANAME BID BNAME 這樣寫錯誤: SQL> select a.id aid,a.name aname,b.id bid,b.name bname from a left join b 因為上面‘二’說過:對 Join 后的數據集不能賦別名,賦別名后提示 Join 操作語法錯誤。 這樣寫也錯誤: SQL> select a.id aid,a.name aname,b.id bid,b.name bname from a
order by和group by都不能出現在where前面。
左外連接 SQL> select a.id aid,a.name aname,b.id bid,b.name bname from a left outer join b AID ANAME BID BNAME
2、右連接 SQL> select a.id aid,a.name aname,b.id bid,b.name bname from a right join b AID ANAME BID BNAME 右連接:以b表為基礎,首先取出b表中所有數據,然后再加上與a,b匹配的的數據
右外連接 SQL> select a.id aid,a.name aname,b.id bid,b.name bname from a right outer join b on a.id=b.id order by b.id; AID ANAME BID BNAME
3、內連接 SQL> select a.id aid,a.name aname,b.id bid,b.name bname from a inner join b AID ANAME BID BNAME
內連接:不以哪個表為基礎,僅取出匹配的數據 可以使用多個join連接多個表,在oracle中多個join不用加括號 FROM t98_indpty_prod_stat T98 但是千萬要注意,from子句中的別名,不支持as,select子句支持as 例子: SQL> select x.id as hh from a x; HH
SQL> select x.id as hh from a as x;
|
SQL UNION 和 UNION ALL 操作符
SQL UNION 操作符
UNION 操作符用於合並兩個或多個 SELECT 語句的結果集。
請注意,UNION 內部的 SELECT 語句必須擁有相同數量的列。列也必須擁有相似的數據類型。同時,每條 SELECT 語句中的列的順序必須相同。
SQL UNION 語法
SELECT column_name(s) FROM table_name1 UNION SELECT column_name(s) FROM table_name2
注釋:默認地,UNION 操作符選取不同的值。如果允許重復的值,請使用 UNION ALL。
SQL UNION ALL 語法
SELECT column_name(s) FROM table_name1 UNION ALL SELECT column_name(s) FROM table_name2
另外,UNION 結果集中的列名總是等於 UNION 中第一個 SELECT 語句中的列名。
下面的例子中使用的原始表:
Employees_China:
E_ID | E_Name |
---|---|
01 | Zhang, Hua |
02 | Wang, Wei |
03 | Carter, Thomas |
04 | Yang, Ming |
Employees_USA:
E_ID | E_Name |
---|---|
01 | Adams, John |
02 | Bush, George |
03 | Carter, Thomas |
04 | Gates, Bill |
使用 UNION 命令
實例
列出所有在中國和美國的不同的雇員名:
SELECT E_Name FROM Employees_China UNION
SELECT E_Name FROM Employees_USA
結果
E_Name |
---|
Zhang, Hua |
Wang, Wei |
Carter, Thomas |
Yang, Ming |
Adams, John |
Bush, George |
Gates, Bill |
注釋:這個命令無法列出在中國和美國的所有雇員。在上面的例子中,我們有兩個名字相同的雇員,他們當中只有一個人被列出來了。UNION 命令只會選取不同的值。
UNION ALL
UNION ALL 命令和 UNION 命令幾乎是等效的,不過 UNION ALL 命令會列出所有的值。
SQL Statement 1 UNION ALL SQL Statement 2
使用 UNION ALL 命令
實例:
列出在中國和美國的所有的雇員:
SELECT E_Name FROM Employees_China UNION ALL
SELECT E_Name FROM Employees_USA
結果
E_Name |
---|
Zhang, Hua |
Wang, Wei |
Carter, Thomas |
Yang, Ming |
Adams, John |
Bush, George |
Carter, Thomas |
Gates, Bill |
來源:http://blog.sina.com.cn/s/blog_61c006ea0100l88i.html