数据库中子查询,联结查询,组合查询的用法和异同

子查询

我们一般情况下会把数据分解后分别存储在不同的表中,使得数据库的表满足第三范式,避免存取重复的数据,但是在查询过程中,我们需要一次性查询不同的表,相关或者无关的数据,这时候就需要使用到联结查询或者子查询。我们先来看看什么是子查询。
子查询是嵌套在另一个查询语句中的查询。
让我们看一个查询的样例:

SELECT * FROM table1 WHERE column1 = (SELECT column1 FROM table2);

上面的例子我们可以看到SELECT * FROM table1 是外层的查询,SELECT column1 FROM table2是子查询。

子查询的好处:

  • 可以在查询过程中以其他表的相关内容作为条件。
  • 比起复杂的联结查询和组合查询提供了可替代的方法。
  • 子查询相对于联结查询和组合查询更容易理解,有比较清晰的结构。
    例子:
    有下面三个表:
    学生表:


    image.png

    成绩表:


    image.png

    学科表:
    image.png

    要求:
    查找年龄为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 = "小花";

结果:


image.png

外联结

外联结有左外联结和右外联结

  • 左外联结
    左外联结将左表作为主表,如果联结的两个表的连接条件匹配,则此行会包含在结果集中,如果左表(主表)中的行与右边中的行不匹配,则会将左边中的行放在结果集中并将右边中的字段都由null填充放入该行
    例如:
    查询所有学生学科成绩信息:
  1. 当学生表为主表:
SELECT * FROM student LEFT JOIN score ON student.id = score.student_id;

结果:


image.png
  1. 当score表为主表:
SELECT * FROM score LEFT JOIN student ON student.id = score.student_id;

结果:


image.png

组合查询

组合查询将多个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;

结果:


image.png

你可能感兴趣的:(数据库中子查询,联结查询,组合查询的用法和异同)