子查询
我们一般情况下会把数据分解后分别存储在不同的表中,使得数据库的表满足第三范式,避免存取重复的数据,但是在查询过程中,我们需要一次性查询不同的表,相关或者无关的数据,这时候就需要使用到联结查询或者子查询。我们先来看看什么是子查询。
子查询是嵌套在另一个查询语句中的查询。
让我们看一个查询的样例:
SELECT * FROM table1 WHERE column1 = (SELECT column1 FROM table2);
上面的例子我们可以看到SELECT * FROM table1 是外层的查询,SELECT column1 FROM table2是子查询。
子查询的好处:
- 可以在查询过程中以其他表的相关内容作为条件。
- 比起复杂的联结查询和组合查询提供了可替代的方法。
-
子查询相对于联结查询和组合查询更容易理解,有比较清晰的结构。
例子:
有下面三个表:
学生表:
成绩表:
学科表:
要求:
查找年龄为18的学生的语文成绩:
SELECT score from score where student_id = (select id from student where age = 18) and subject_id = (select id from subject where subject = "语文");
结果:
但是子查询也有相应的缺点:
- 当查询的表过多,嵌套的层数变多都会使查询语句变得复杂,可读性也会变差。
- 子查询的结构是从里层到外层,里层的结果在语句的其他部分不可用,性能较差,查询范围也会受限。
联结查询
看了上面的子查询后再来了解另外一种可以进行多表查询的方式:
联结查询可以代替子查询进行跨表查询,效率会明显提高。
我们看一下联结查询的不同形式:
内联结:
内联结是将一个表中的行与其他表中的行进行匹配,得到两个表的交集。
内联结的语法如下:
SELECT column_list FROM table1
INNER JOIN t2 ON join_condition1
(INNER JOIN t3 ON join_condition2)
...WHERE conditions
例如联结学生表和成绩表,查出姓名为小花的成绩信息:
SELECT * FROM score INNER JOIN student ON score.student_id = student.id WHERE student.name = "小花";
结果:
外联结
外联结有左外联结和右外联结
- 左外联结
左外联结将左表作为主表,如果联结的两个表的连接条件匹配,则此行会包含在结果集中,如果左表(主表)中的行与右边中的行不匹配,则会将左边中的行放在结果集中并将右边中的字段都由null填充放入该行
例如:
查询所有学生学科成绩信息:
- 当学生表为主表:
SELECT * FROM student LEFT JOIN score ON student.id = score.student_id;
结果:
- 当score表为主表:
SELECT * FROM score LEFT JOIN student ON student.id = score.student_id;
结果:
组合查询
组合查询将多个select语句中的两个或者多个结果集合并到一个结果集中。
组合查询语法:
SELECT column_list
UNION [DISTINCT | ALL]
SELECT column_list
UNION [DISTINCT | ALL]
...
要使用UNION运算符组合两个或多个查询结果的结果集时必须满足以下条件:
- 所有SELECT语句中出现的列的数量和顺序都必须相同。
- 列的数据类型必须相同或者可转换
默认情况下,如果未指定运算符,UNION运算符也会删除重复的行。
比如:
将custom表和employee表组合起来,custom和employee表都只有id,name两个字段。
CREATE TABLE employee (
id INT NOT NULL,
name VARCHAR(100) NOT NULL
)ENGINE=InnoDB DEFAULT CHARSET=utf8;
INSERT INTO employee VALUES(1,"Molly"),(2,"Micheal"),(3,"John");
CREATE TABLE custom (
id INT NOT NULL,
name VARCHAR(100) NOT NULL
)ENGINE=InnoDB DEFAULT CHARSET=utf8;
INSERT INTO custom VALUES(12,"Mick"),(16,"Susan");
组合查询:
SELECT id,name FROM employee UNION SELECT id,name FROM custom;
结果: