sql 表之间连接方式

1.今天看到一条sql语句select * from a表,b表

什么结果都没有,愣是没看出有错,

查了一下默认是笛卡尔积a表有数据,b表数据为零,a×0=0肯定没有数据。

而全外连接 full join 才是全显示,没有数据的一边填充null。

 

2.网上的一个示例:有东西防止忘了后重拾太慢。

   联接可分为以下几类:     
    
   内联接(典型的联接运算,使用像    =    或    <>    之类的比较运算符)。包括相等联接和自然联接。     
   内联接使用比较运算符根据每个表共有的列的值匹配两个表中的行。例如,检索    students    和    courses    表中学生标识号相同的所有行。   
    
   外联接。外联接可以是左向外联接、右向外联接或完整外部联接。     
   在    FROM    子句中指定外联接时,可以由下列几组关键字中的一组指定:   
    
   LEFT    JOIN    或    LEFT    OUTER    JOIN。     
   左向外联接的结果集包括    LEFT    OUTER    子句中指定的左表的所有行,而不仅仅是联接列所匹配的行。如果左表的某行在右表中没有匹配行,则在相关联的结果集行中右表的所有选择列表列均为空值。   
    
   RIGHT    JOIN    或    RIGHT    OUTER    JOIN。     
   右向外联接是左向外联接的反向联接。将返回右表的所有行。如果右表的某行在左表中没有匹配行,则将为左表返回空值。   
    
   FULL    JOIN    或    FULL    OUTER    JOIN。     
   完整外部联接返回左表和右表中的所有行。当某行在另一个表中没有匹配行时,则另一个表的选择列表列包含空值。如果表之间有匹配行,则整个结果集行包含基表的数据值。   
    
   交叉联接。     
   交叉联接返回左表中的所有行,左表中的每一行与右表中的所有行组合。交叉联接也称作笛卡尔积。  

 
     仅当至少有一个同属于两表的行符合联接条件时,内联接才返回行。内联接消除与另一个表中的任何行不匹配的行。而外联接会返回    FROM    子句中提到的至少一个表或视图的所有行,只要这些行符合任何    WHERE    或    HAVING    搜索条件。将检索通过左向外联接引用的左表的所有行,以及通过右向外联接引用的右表的所有行。完整外部联接中两个表的所有行都将返回。   
   例如,下面的内联接检索与某个出版商居住在相同州和城市的作者:   
    
   USE    pubs   
   SELECT    a.au_fname,    a.au_lname,    p.pub_name   
   FROM    authors    AS    a    INNER    JOIN    publishers    AS    p   
         ON    a.city    =    p.city   
         AND    a.state    =    p.state   
   ORDER    BY    a.sau_lname    ASC,    a.au_fname    ASC   
    
   FROM    子句中的表或视图可通过内联接或完整外部联接按任意顺序指定;但是,用左或右向外联接指定表或视图时,表或视图的顺序很重要。有关使用左或右向外联接排列表的更多信息,请参见使用外联接。     

 

3.示例:

假设有如下表:


一个为投票主表,一个为投票者信息表~记录投票人IP及对应投票类型,左右连接实际说是我们联合查询的结果以哪个表为准~
1:如右接连 right join 或 right outer join:

我们以右边voter表为准,则左表(voteMaster)中的记录只有当其ID在右边(voter)中存在时才会显示出来,如上图,左边中ID为3.4.5.6因为这些ID右表中没有相应记录,所以没有显示!
2:因此我们自然能理解左连接 left join 或者 left outer join 

可见,现在右边中ID在中存在时才会显示,当右边中没有相应数据时则用NULL代替!
3:全连接 full join 或者 full outer join,为二个表中的数据都出来,这里演示效果与上一样! 
4:内连接 inner join 或者 join;它为返回字段ID同时存在于表voteMaster 和 voter中的记录

5:交叉连接(完全连接)cross join 不带 where 条件的
没有 WHERE 子句的交叉联接将产生联接所涉及的表的笛卡尔积。第一个表的行数乘以第二个表的行数等于笛卡尔积结果集的大小。(table1和table2交叉连接产生 6*3=18条记录)

等价select vm.id,vm.voteTitle,vt.ip from voteMaster as vm,voter as vt

6:自连接。在这里我用我前段时间一个电力项目中的例子(改造过)
如下表:

这是一个部门表,里面存放了部门及其上级部门,但都放在同一张表中,我们假设现在需要用SQL查询出各部门及其上级部门!就如何做,
当然,不用自连接也一样,可以如下:

我们达到预期目的!在这个查询中使用了一个子查询完成对上级部门名的查询,如果使用自连接,那么结构上感觉会清晰很多。

是不是也同样完成了功能呢,这里除了使用自连接外,还使用了左连接,因为省电力没有上级部门,他是老大,如果使用内连接,就会把这条记录过滤掉,因为没有和他匹配的上级部门。
自连接用的比较多的就是对权形结构的查询!类似上表






SQL 表连接

通俗的讲:  
  A   
left   join   B   的连接的记录数与A表的记录数同  
  A   
right   join   B   的连接的记录数与B表的记录数同    
  A   
left   join   B   等价B   right   join   A  
   
   
  
table   A:  
Field_K,   Field_A  
  1                       a  
  
3                       b  
  
4                       c  
   
  
table   B:  
  Field_K,   Field_B  
  
1                       x  
  
2                       y  
  
4                       z  
   
  
select   a.Field_K,   a.Field_A,   b.Field_K,   b.Field_B  
  
from   a   left   join   b   on   a.Field_K=b.Field_K  
   
  Field_K         Field_A         Field_K         Field_B          
  
----------   ----------   ----------   ----------    
  1                     a                     1                     x                    
  
3                     b                     NULL               NULL  
  
4                     c                     4                     z                    
   
  
select   a.Field_K,   a.Field_A,   b.Field_K,   b.Field_B  
  
from   a   right   join   b   on   a.Field_K=b.Field_K  
   
  Field_K         Field_A         Field_K         Field_B          
  
----------   ----------   ----------   ----------    
  1                     a                     1                     x                    
  
NULL               NULL               2                     y                    
  
4                     c                     4                     z       --

举个例子:  
  假设a表和b表的数据是这样的。  
  a                         b    
  id     name  id     stock   
  
1  a             1         15  
  
2         b             2         50  
  
3         c                  
   
  
select   *   from   a   inner   join   b   on   a.id=b.id  
  这个语法是连接查询中的内连接,它产生的结果是  
  两个表相匹配的记录出现在结果列表中。  
  根据上面的表,出现的结果是这样的  
  a.id     name     b.id     stock  
  
1       a             1         15  
  
2             b             2         50  
  
----------------------------  
  select   *   from   a,b   where   a.id=b.id  
  这个语法是内连接的另外一种写法,其执行结果与inner   
join   一样  
   
  
--------------------------------    
   
  
select   *   from   a   left/right   join   b   on   a.id=b.id  
  这个是外连接语法中的左外连接或右外连接  
  如果是左外连接的话,它将显示a表的所有记录,  
  
select   a.*,b.*   from   a   left   join   b   on   a.id=b.id  
  查询的结果是这样的:  
  a.id     name     b.id     stock  
  
1         a         1             15  
  
2               b         2             50  
  
3               c       null         null   
  
--------------------------------------------  
  如果是右外连接的话,它将显示b表的所有记录,  
  
select   a.*,b.*   from   a   right   join   b   on   a.id=b.id  
  查询的结果是这样的:  
  a.id     name     b.id     stock  
  
1         a         1             15  
  
2               b         2             50   

总结:left join 和right join相当于表字段的扩展,多加了另一个表的字段,,left表示本表扩展,right表示参考表扩展

--


阅读更多
个人分类: db数据库
想对作者说点什么? 我来说一句

没有更多推荐了,返回首页

加入CSDN,享受更精准的内容推荐,与500万程序员共同成长!
关闭
关闭