有许多演示如何使用MySQL所支持的连接操作的示例,都用到了下列两个表 t1 和 t2:

mysql> SELECT * FROM t1;
+----+------+
| i1 | c1 |
+----+------+
| 1 | a |
| 2 | b |
| 3 | c |
+----+------+ mysql> SELECT * FROM t2;
+----+------+
| i2 | c2 |
+----+------+
| 2 | c |
| 3 | b |
| 4 | a |
+----+------+

  由于这两个表都很小,因此我们可以很清楚地看到每种类型的连接操作所执行的效果。

1.内连接

1.1  如果SELECT 语句的 FROM 子句里列出了过个表的名字,并且这些名字是由 INNER JOIN 隔开的,那么 MySQL 就会执行一个内连接(inner join)操作----它会把一个表里的行与另一个表里的行进行匹配,然后产生结果。例如,如果像下面这样把 t1 和 t2 连接起来,那么 t1 里的每一行将于 t2 里的每一行进行组合:

mysql> SELECT * FROM t1 INNER JOIN t2;
+----+------+----+------+
| i1 | c1 | i2 | c2 |
+----+------+----+------+
| 1 | a | 2 | c |
| 2 | b | 2 | c |
| 3 | c | 2 | c |
| 1 | a | 3 | b |
| 2 | b | 3 | b |
| 3 | c | 3 | b |
| 1 | a | 4 | a |
| 2 | b | 4 | a |
| 3 | c | 4 | a |
+----+------+----+------+

  上面这条语句也可以写成:

mysql> SELECT t1.* ,t2.* FROM t1 INNER JOIN t2;
+----+------+----+------+
| i1 | c1 | i2 | c2 |
+----+------+----+------+
| 1 | a | 2 | c |
| 2 | b | 2 | c |
| 3 | c | 2 | c |
| 1 | a | 3 | b |
| 2 | b | 3 | b |
| 3 | c | 3 | b |
| 1 | a | 4 | a |
| 2 | b | 4 | a |
| 3 | c | 4 | a |
+----+------+----+------+

  根据某个表里的每一行与另一个表里的每一行得到的全部可能组合的连接操作,叫做生成 '笛卡尔积'(cartesian product)。按照这种方式连接表,很可能会产生数量巨大的行。如果增加 WHERE 子句,实现表之间基于某些列值的匹配,那么整个连接操作只会选取那些列值彼此相等的行:

mysql> SELECT t1.* , t2.* FROM t1 INNER JOIN t2 WHERE t1.i1=t2.i2;
+----+------+----+------+
| i1 | c1 | i2 | c2 |
+----+------+----+------+
| 2 | b | 2 | c |
| 3 | c | 3 | b |
+----+------+----+------+

连接类型 CROSS JOIN 和 JOIN 都等同于 INNER JOIN ,因此下面的语句时等价的:

SELECT t1.* , t2.* FROM t1 INNER JOIN t2 WHERE t1.i1=t2.i2;

SELECT t1.* , t2.* FROM t1 CROSS JOIN t2 WHERE t1.i1=t2.i2;

SELECT t1.* , t2.* FROM t1 JOIN t2 WHERE t1.i1=t2.i2;

  

  在指定如何对表里的列进行匹配方面,INNER JOIN、CROSS JOIN 和 JOIN (未包括逗号运算符) 还支持另外几种语法格式:

  (i) 有一种语法是用 ON 子句代替 WHERE 子句。下面示例使用 INNER JOIN 展示了这一点:

SELECT  t1.* , t2.* FROM t1 INNER JOIN t2 ON t1.i1=t2.i2;

  不管被连接的列是否同名,都可以使用ON。

  (ii) 另一种语法是使用 USING() 子句。它在概念上类似于 ON 子句,但要求被连接的列必须同名。例如,下面这条查询语句可以将 mytbl1.b 连接到 mytbl2.b:

SELECT mytbl1.* , mytbl2.* FROM mytbl1 INNER JOIN mytbl2 USING(b);

1.2 对被连接的表里的列引用进行限定

  如果被连接的表具有相同名字的列 b ,那么引用相同的列时必须将它限定为 mytbl1.b 或 mytbl2.b:

SELECT a, mytbl1.b, mytbl2.b, c FROM mytbl1 INNER JOIN mytbl2 ... ;

  有时候,用表的名字进行限定仍不足以解决列的歧义问题。例如,用户正在进行一个自连接操作(即把一个表与它自身连接起来),那么这会在同一条查询命令里多次用到同一个表,此时使用表明来限定列名已经无济于事。在这种情况下,可以使用表的别名来解决问题。只需为该表的某个实例取一个别名,便可以通过 alias_name.col_name 语法来引用该实例里的列。例如:

mysql> SELECT t1.i1, t.c1 from t1 inner join t1 as t
-> where t1.i1 > t.i1;
+----+------+
| i1 | c1 |
+----+------+
| 2 | a |
| 3 | a |
| 3 | b |
+----+------+

2. 左(外)连接和右(外)连接

2.1  内连接只会显示在两个表里都匹配上的行。外连接除了显示同样的匹配结果,还可以把其中一个表在另一个表里没有匹配的行业显示出来。外连接分为 LEFT JOIN 和 RIGHT JOIN。LEFT JOIN 会把左表里的在右表里未匹配上的行也显示出来。RIGHT JOIN 在表的顺序上刚好与之相反。

  LEFT JOIN 的工作方式是:先指定用于对两个表里的行进行匹配的列;接着,当左表的某行与右表的某行相匹配时,两行的内容会被选取为一个输出行;当左表的某行在右表里无匹配时,它仍会被选为一个输出行,只是与它连接的是右表的一个“假”行,其中的每一列都包含“NULL”。

  例如:

mysql> SELECT * FROM t1 LEFT JOIN t2 ON t1.i1=t2.i2;
+----+------+------+------+
| i1 | c1 | i2 | c2 |
+----+------+------+------+
| 2 | b | 2 | c |
| 3 | c | 3 | b |
| 1 | a | NULL | NULL |
+----+------+------+------+

  注意:在使用 LEFT JOIN 时,只有将右表里的列全部定义成 NOT NULL ,结果集里的行才会没有问题。例如,右表里包含值为 NULL 的列,那么你就无法将这些 NULL 值与标识为匹配行的那些 NULL 值区别开来。

  RIGHT JOIN 与 LEFT JOIN 很相似,只是表的角色需要对换一下:

mysql> SELECT * FROM t1 RIGHT JOIN t2 ON t1.i1=t2.i2;
+------+------+----+------+
| i1 | c1 | i2 | c2 |
+------+------+----+------+
| 2 | b | 2 | c |
| 3 | c | 3 | b |
| NULL | NULL | 4 | a |
+------+------+----+------+

2.2 当只想将左表里的那些在右表里无匹配的行找出来时,LEFT JOIN 非常有用。要实现这个功能,只需增加一条 WHERE 子句,让它把右表列值为 NULL 的行筛选出来:

mysql> SELECT * FROM t1 LEFT JOIN t2 ON t1.i1=t2.i2
-> WHERE t2.i2 IS NULL;
+----+------+------+------+
| i1 | c1 | i2 | c2 |
+----+------+------+------+
| 1 | a | NULL | NULL |
+----+------+------+------+

  上例中,把来自右表值为 NULL 的列显示出来毫无意义,因此可以把它们从输出列的列表里省略掉:

mysql> SELECT t1.i1 FROM t1 LEFT JOIN t2 ON t1.i1=t2.i2 WHERE t2.i2 IS NULL;
+----+
| i1 |
+----+
| 1 |
+----+

MySQL数据库(9)----使用连接实现多表检索的更多相关文章

  1. mysql数据库可以远程连接或者说用IP地址可以访问

    mysql数据库可以远程连接或者说用IP地址可以访问 一般情况不建议直接修改root的权限, 先看下,自己mysql数据库的用户级权限 mysql -u root -p----->用root登陆 ...

  2. MySQL数据库无法远程连接的解决办法

    远程登陆数据库的时候出现了下面出错信息: ERROR 2003 (HY000): Can't connect to MySQL server on 'xxx.xxx.xxx.xxx', 经过今天下午的 ...

  3. 在mysql数据库中制作千万级测试表

    在mysql数据库中制作千万级测试表 前言: 最近准备深入的学一下mysql,包括各种引擎的特性.性能优化.分表分库等.为了方便测试性能.分表等工作,就需要先建立一张比较大的数据表.我这里准备先建一张 ...

  4. mysql数据库delete数据时不支持表别名!!!

    mysql数据库delete数据时不支持表别名!!! mysql delete时候 提示语法错误!如下sql: 去掉 表别名的时候: 正确的写法例如: DELETE FROM COMMENTS_REP ...

  5. MYSQL数据库类型与JAVA类型对应表

    MYSQL数据库类型与JAVA类型对应表   MYSQL数据库类型与JAVA类型对应表 类型名称 显示长度 数据库类型 JAVA类型 JDBC类型 索引(int) VARCHAR L+N VARCHA ...

  6. MySQL 数据库实现远程连接

    1,刚开始我使用的是Navicat for MySQL工具连接远程的mysql的数据库. 报错了.报错信息是 Error 1130: Host '192.168.1.80' is not allowe ...

  7. 解决Mysql数据库拒绝远程连接和忘记密码的问题

    解决数据库忘记密码的问题 ERROR 1045 (28000): Access denied for user 'root'@'localhost' (using password: YES) 编辑m ...

  8. mysql数据库允许远程连接

    1.验证初始是否允许远程连接 由于本次虚拟机IP为192.168.2.120,因此我们执行 mysql -h 192.168.20.120 -P 3306 -u root -proot(备注:-pro ...

  9. 用C语言操作MySQL数据库,进行连接、插入、修改、删除等操作

    C/C++ code   ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 3 ...

随机推荐

  1. [转] 2018年最新桌面CPU性能排行天梯图(含至强处理器)

    [FROM] http://www.idn100.com/zuzhuangdiannaopeizhi-pc2849/ 排名 处理器 图例 分数 1 Intel Xeon Platinum 8173M ...

  2. Scrapyd API的安装

    安装好了Scrapyd之后,我们可以直接请求它提供的API来获取当前主机的Scrapy任务运行状况.比如,某台主机的IP为192.168.1.1,则可以直接运行如下命令获取当前主机的所有Scrapy项 ...

  3. 思科 ISR路由器登录内置交换模块的方式

    ISR2900/3900系列 登录:Router#service-module gigabitethernet1/0 session 退出: control+shift+6 x disconnect ...

  4. (转)搞个这样的APP要多久?心酸啊。

    这是一个“如有雷同,纯属巧合”的故事,外加一些废话,大家请勿对号入座.开始了…… 我有些尴尬地拿着水杯,正对面坐着来访的王总,他是在别处打拼的人,这几年据说收获颇丰,见移动互联网如火如荼,自然也想着要 ...

  5. 【Kafka】Kafka集群搭建

    一.准备工作 服务器:最好是多台,大于等于2 已经搭建好的zookeeper集群 下载软件kafka_2.11-0.10.0.1.tgz 二.创建目录 #创建目录 cd /opt/ mkdir kaf ...

  6. android 签名验证防止重打包

    网上资料很多,这里只做一个笔记反编译 dex 修改重新打包签名后 apk 的签名信息肯定会改变,所以可以在代码中判断签名信息是否被改变过,如果签名不一致就退出程序,以防止 apk 被重新打包. 1 j ...

  7. SpringMVC 的工作机制

    在一个工程中如果想要使用 SpringMVC的话,只需要两个步骤 在web.xml中配置一个DispatcherServlet.需要配置一个org.springframework.web.servle ...

  8. APU (美国AMD公司研发的加速处理器)

    APU(Accelerated Processing Unit)中文名字叫加速处理器,是AMD“融聚未来”理念的产品,它第一次将中央处理器和独显核心做在一个晶片上,它同时具有高性能处理器和最新独立显卡 ...

  9. 原生javascript实现分页效果+搜索功能

    一.概述 首先,我们要明确为何需要分页技术,主要原因有以下: 1.分页可以提高客户体验度,适当地选择合适的数据条数,让页面显得更有条理,使得用户体验感良好,避免过多数据的冗余. 2.提高性能的需要. ...

  10. OpenTLD在VS2012和opencv246编译通过

    最近看到了TLD的跟踪视频,觉得很有意思,刚好最近在看行人检测所以就打算下载源码玩一玩,因为源码是Linux版本的(原作者写的是C++和MATLAB的混合编程)C++源码可以在我的博客TLD(一种目标 ...