数据库——MySQL——多表查询

这里多表,为了方便我只建了两张表,更复杂的表间也就是这些东西,就是复杂程度不一样。

数据源准备

建立一个学生表,和一个班级表

# 建立学生表
create table student(
id int not null unique auto_increment,
name varchar(20) not null,
sex enum('male','female') not null default 'male',
age int(3) unsigned not null default 28,
birthday date not null,
class_id int
);

# 建立班级表
create table class(
id int not null unique auto_increment,
class_name varchar(20) not null
);
建立表
# 给学生表插入数据
insert into student(name,sex,age,birthday,class_id) values
('成龙','male',48,'20101111',1),
('歪歪','female',48,'20150311',2),
('丫丫','female',38,'20101101',2),
('丁丁','female',18,'20110312',2),
('星星','female',18,'20160513',2),
('格格','female',28,'20170127',2),
('张野','male',28,'20160311',3),
('程咬金','male',18,'19970312',3),
('程咬银','female',18,'20130311',3),
('程咬铜','male',18,'20150411',3),
('程咬铁','female',18,'20140512',3);

# 给班级表插入数据
insert into class (class_name) values
 ("一班"),
 ("二班");
插入数据

多表连接查询

交叉连接

不适用任何匹配条件,生成笛卡儿积。

交叉连接生成的笛卡儿积是最全的数据源,但是基本无用,因为大量无用的数据充斥其中。所以我们需要加入条件,找到我们需要的数据。

 内连接

只连接匹配的行

找两张表共有个部分,相当于利用条件从笛卡儿积结果中筛选正确的结果。

mysql> mysql> select * from student, class where student.class_id = class.id;
+----+--------+--------+-----+------------+----------+----+------------+
| id | name   | sex    | age | birthday   | class_id | id | class_name |
+----+--------+--------+-----+------------+----------+----+------------+
|  1 | 成龙   | male   |  48 | 2010-11-11 |        1 |  1 | 一班       |
|  2 | 歪歪   | female |  48 | 2015-03-11 |        2 |  2 | 二班       |
|  3 | 丫丫   | female |  38 | 2010-11-01 |        2 |  2 | 二班       |
|  4 | 丁丁   | female |  18 | 2011-03-12 |        2 |  2 | 二班       |
|  5 | 星星   | female |  18 | 2016-05-13 |        2 |  2 | 二班       |
|  6 | 格格   | female |  28 | 2017-01-27 |        2 |  2 | 二班       |
+----+--------+--------+-----+------------+----------+----+------------+

因为我在班级表中没有定义三班,所以筛选的查询结果中没有找到三班和三班人员的记录。

除了上面的的写法外,还有一种写法

select * from student inner join class on student.class_id=class.id;

两个语句查找的内容是一致的。

外连接

外连接之左连接:优先显示左表的全部记录

本质上就是在内连接的基础上增加左边有,右边没有的结果

mysql> select * from student left join class on student.class_id=class.id; 
+----+-----------+--------+-----+------------+----------+------+------------+ | id | name | sex | age | birthday | class_id | id | class_name | +----+-----------+--------+-----+------------+----------+------+------------+ | 1 | 成龙 | male | 48 | 2010-11-11 | 1 | 1 | 一班 | | 2 | 歪歪 | female | 48 | 2015-03-11 | 2 | 2 | 二班 | | 3 | 丫丫 | female | 38 | 2010-11-01 | 2 | 2 | 二班 | | 4 | 丁丁 | female | 18 | 2011-03-12 | 2 | 2 | 二班 | | 5 | 星星 | female | 18 | 2016-05-13 | 2 | 2 | 二班 | | 6 | 格格 | female | 28 | 2017-01-27 | 2 | 2 | 二班 | | 7 | 张野 | male | 28 | 2016-03-11 | 3 | NULL | NULL | | 8 | 程咬金 | male | 18 | 1997-03-12 | 3 | NULL | NULL | | 9 | 程咬银 | female | 18 | 2013-03-11 | 3 | NULL | NULL | | 10 | 程咬铜 | male | 18 | 2015-04-11 | 3 | NULL | NULL | | 11 | 程咬铁 | female | 18 | 2014-05-12 | 3 | NULL | NULL | +----+-----------+--------+-----+------------+----------+------+------------+ 11 rows in set (0.00 sec)

外连接之右连接,优先显示右表的全部记录

本质上就是在内连接的基础上加上右边有,左边没有的结果

mysql> select * from student right join class on student.class_id=class.id;
+------+--------+--------+------+------------+----------+----+------------+
| id   | name   | sex    | age  | birthday   | class_id | id | class_name |
+------+--------+--------+------+------------+----------+----+------------+
|    1 | 成龙   | male   |   48 | 2010-11-11 |        1 |  1 | 一班       |
|    2 | 歪歪   | female |   48 | 2015-03-11 |        2 |  2 | 二班       |
|    3 | 丫丫   | female |   38 | 2010-11-01 |        2 |  2 | 二班       |
|    4 | 丁丁   | female |   18 | 2011-03-12 |        2 |  2 | 二班       |
|    5 | 星星   | female |   18 | 2016-05-13 |        2 |  2 | 二班       |
|    6 | 格格   | female |   28 | 2017-01-27 |        2 |  2 | 二班       |
+------+--------+--------+------+------------+----------+----+------------+
6 rows in set (0.00 sec)

 

全外连接,显示左右两个表中全部记录

全连接本质上就是在内连接的基础上增加左边有,右边没有的,和右边没有,左边没有的结果。

然而在mysql中是没有full join的,所以,我们需要用union。

mysql> select * from student left join class on student.class_id=class.id
    -> union
    -> select * from student right join class on student.class_id=class.id;
+------+-----------+--------+------+------------+----------+------+------------+
| id   | name      | sex    | age  | birthday   | class_id | id   | class_name |
+------+-----------+--------+------+------------+----------+------+------------+
|    1 | 成龙      | male   |   48 | 2010-11-11 |        1 |    1 | 一班       |
|    2 | 歪歪      | female |   48 | 2015-03-11 |        2 |    2 | 二班       |
|    3 | 丫丫      | female |   38 | 2010-11-01 |        2 |    2 | 二班       |
|    4 | 丁丁      | female |   18 | 2011-03-12 |        2 |    2 | 二班       |
|    5 | 星星      | female |   18 | 2016-05-13 |        2 |    2 | 二班       |
|    6 | 格格      | female |   28 | 2017-01-27 |        2 |    2 | 二班       |
|    7 | 张野      | male   |   28 | 2016-03-11 |        3 | NULL | NULL       |
|    8 | 程咬金    | male   |   18 | 1997-03-12 |        3 | NULL | NULL       |
|    9 | 程咬银    | female |   18 | 2013-03-11 |        3 | NULL | NULL       |
|   10 | 程咬铜    | male   |   18 | 2015-04-11 |        3 | NULL | NULL       |
|   11 | 程咬铁    | female |   18 | 2014-05-12 |        3 | NULL | NULL       |
+------+-----------+--------+------+------------+----------+------+------------+
11 rows in set (0.00 sec)

子查询

子查询是将一个查询语句嵌套在另一个查询语句中。

内层查询语句的结果可以为外层查询语句提供查询条件

子查询中可以包含:in, not in, any, all, exists , not exists,union, union all等关键字。还可以包含:=,!=, >, <等。

# any 关键字
select ...from ... where a > any(...);
相当于
select ...from ... where a > result1 or a > result2 or a > result3;

# all关键字
与any关键字类似,只不过上面的or改成and。即:
select ...from ... where a > all(...);
相当于
select ...from ... where a > result1 and a > result2 and a > result3;

# some关键字
some关键字和any关键字是一样的功能。所以:
select ...from ... where a > some(...);
相当于
select ...from ... where a > result1 or a > result2 or a > result3;

# in关键字
in运算符用于where语句中,以列表项的形式支持多个选择。
select * from student where class_id in (1,3);
select * from student where class_id not in (1,3);
select * from student where class_id in (select id from class);
最后的这个查询语句等价于:select * from student where class_id=any (select id from class);
not in 和in作用相反

# exists关键字
exists只返回True或者False。
select * from student where exists (select * from class);    # exists返回True,外层查询会执行。
select * from student where exists (select * );    # exists返回True,外层查询会执行。
select * from student where exists (select 1);    # exists返回True,外层查询会执行。
select * from student where exists (select * from class where class_id =4);    # exists返回True,外层查询不会执行。
select * from student where sex="male" and exists(select * from class);        # 可以配合其他查询条件一起使用。
not exists和exists作用相反

# union关键字
用来将多个select语句的结果组合到一个结果集中,前面说全外间连接有相应的例子。
在多个select语句中,对应的列应该有相同的字段属性,且第一个select语句中被使用的字段名称也被用于结果的字段名称。
union和union all的区别:当使用union时,MySQL会把结果集中重复的记录删除掉,而使用union all后,mysql会把所有记录返回,而且效率要比union高。

 

posted @ 2018-08-26 16:26  苦行僧95  阅读(346)  评论(0编辑  收藏  举报