sql語句的join用法


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 子句錯誤。
若要創建一個只包括在被聯接字段中具有相同數據的記錄,使用 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;

SQL> select * from a;

        ID       NAME
---------- ----------
         1          1
         2          2
         3          3
         4          4
         5          5
         6          6
         7          7
         8          8
         9          9

SQL> select * from b;

        ID       NAME
---------- ----------
         6          6
         7          7
         8          8
         9          9
        10         10
        11         11
        12         12
        13         13
        14         14

兩個表a,b相連接,要取出id相同的字段,如下:

 

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前面。

 

   左外連接

SQL> select a.id aid,a.name aname,b.id bid,b.name bname from a left outer 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

 

2、右連接

SQL> select a.id aid,a.name aname,b.id bid,b.name bname from a  right join b
  2  on a.id=b.id order by b.id;

       AID      ANAME        BID      BNAME
---------- ---------- ---------- ----------
         6          6          6          6
         7          7          7          7
         8          8          8          8
         9          9          9          9
                              10         10
                              11         11
                              12         12
                              13         13
                              14         14

右連接:以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
---------- ---------- ---------- ----------
         6          6          6          6
         7          7          7          7
         8          8          8          8
         9          9          9          9
                              10         10
                              11         11
                              12         12
                              13         13
                              14         14

 

3、內連接

SQL> select a.id aid,a.name aname,b.id bid,b.name bname from a inner join b
  2  on a.id=b.id order by a.id;

       AID      ANAME        BID      BNAME
---------- ---------- ---------- ----------
         6          6          6          6
         7          7          7          7
         8          8          8          8
         9          9          9          9

 

內連接:不以哪個表為基礎,僅取出匹配的數據
 

可以使用多個join連接多個表,在oracle中多個join不用加括號

  FROM t98_indpty_prod_stat   T98
  LEFT JOIN t02_prod_group_rela_h   T02
    ON T98.Product_Id = T02.Product_Id
  LEFT JOIN t99_product_grp_rela_cd  T99
    ON T02.Product_Group_id = T99.Product_Grp_Fouth_Lvl_Cd
  WHERE T98.Summ_Date = '2010-02-28'
  AND T98.Txdate = '2010-02-20'
  GROUP BY t98.Summ_Date,t98.Stat_Org_Id,t98.Conform_Indparty_Id,t98.Province_Cd,t98.Txdate;

但是千萬要注意,from子句中的別名,不支持as,select子句支持as

 例子:

SQL> select x.id as hh from a x;

        HH
----------
         1
         2
         3
         4
         5
         6
         7
         8
         9

 

SQL> select x.id as hh from a as x;
select x.id as hh from a as x
                         *
ERROR at line 1:
ORA-00933: SQL command not properly ended

 

-----------------[以下為網上的一點資料]---------------

LEFT JOIN操作用於在任何的 FROM 子句中,組合來源表的記錄。使用 LEFT JOIN 運算來創建一個左邊外部聯接。左邊外部聯接將包含了從第一個(左邊)開始的兩個表中的全部記錄,即使在第二個(右邊)表中並沒有相符值的記錄。

 

不同的 SQL JOIN

除了我們在上面的例子中使用的 INNER JOIN(內連接),我們還可以其他幾種連接。

下面列出了您可以使用的 JOIN 類型,以及它們之間的差異。

  • JOIN: 如果表中有至少一個匹配,則返回行,join和inner join相同。 
  • LEFT JOIN: 即使右表中沒有匹配,也從左表返回所有的行
  • RIGHT JOIN: 即使左表中沒有匹配,也從右表返回所有的行
  • FULL JOIN: 只要其中一個表中存在匹配,就返回行

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

 

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

 

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM