MySQL: 多表查询

1 什么是多表查询

    DQL: 查询多张表, 获取到需要的数据

    比如: 查询家电分类下, 都有哪些商品, 那么, 我们就需要查询分类与商品这两张表

2 数据准备

    1) 创建 db3_2 数据库

-- 创建 db3_2 数据库,指定编码 
CREATE DATABASE db3_2 CHARACTER SET utf8;

    2) 创建分类表与商品表

    

#分类表 (一方 主表) 
CREATE TABLE category (
  cid VARCHAR(32) PRIMARY KEY ,
  cname VARCHAR(50) 
);

#商品表 (多方 从表) 
CREATE TABLE products(
    pid VARCHAR(32) PRIMARY KEY ,
    pname VARCHAR(50),
    price INT,
    flag VARCHAR(2),     #是否上架标记为:1表示上架、0表示下架 
    category_id VARCHAR(32),
    -- 添加外键约束
    FOREIGN KEY (category_id) REFERENCES category (cid) 
);

   3) 插入数据

#分类数据 
INSERT INTO category(cid,cname) VALUES('c001','家电'); 
INSERT INTO category(cid,cname) VALUES('c002','鞋服'); 
INSERT INTO category(cid,cname) VALUES('c003','化妆品'); 
INSERT INTO category(cid,cname) VALUES('c004','汽车');

#商品数据
INSERT INTO products(pid, pname,price,flag,category_id) VALUES('p001','小米电视机',5000,'1','c001'); 
INSERT INTO products(pid, pname,price,flag,category_id) VALUES('p002','格力空调',3000,'1','c001'); 
INSERT INTO products(pid, pname,price,flag,category_id) VALUES('p003','美的冰箱',4500,'1','c001');
INSERT INTO products (pid, pname,price,flag,category_id) VALUES('p004','篮球鞋',800,'1','c002'); 
INSERT INTO products (pid, pname,price,flag,category_id) VALUES('p005','运动裤',200,'1','c002');
INSERT INTO products (pid, pname,price,flag,category_id) VALUES('p006','T 恤',300,'1','c002');
INSERT INTO products (pid, pname,price,flag,category_id) VALUES('p007','冲锋衣',2000,'1','c002');
INSERT INTO products (pid, pname,price,flag,category_id) VALUES('p008','神仙水',800,'1','c003'); 
INSERT INTO products (pid, pname,price,flag,category_id) VALUES('p009','大宝',200,'1','c003');

3 笛卡尔积 

交叉连接查询,因为会产生笛卡尔积,所以基本不会使用

1) 语法格式

SELECT 字段名 FROM  表1, 表2;

2) 使用交叉连接查询 商品表与分类表

SELECT * FROM category , products;

3) 观察查询结果,产生了笛卡尔积 (得到的结果是无法使用的)

 4) 笛卡尔积图解    

    假设集合A={a, b},集合B={0, 1, 2},则两个集合的笛卡尔积为{(a, 0), (a, 1), (a, 2), (b, 0), (b, 1), (b, 2)}。

    

4 多表查询 -  内连接查询 

<1>  内连接的特点

      通过指定的条件去匹配两张表中的数据, 匹配上就显示,匹配不上就不显示      

      比如通过: 从表的外键 = 主表的主键 的方式去匹配

<2>  隐式内连接

      from子句后面直接写多个表名, 使用where指定连接条件。这种连接方式,是 隐式内连接。

      (使用where条件过滤无用的数据)

    语法格式  

SELECT 字段名 FROM 左表, 右表 WHERE 连接条件;

代码示例:

1) 查询所有商品的所有信息,和对应的分类信息

# 隐式内连接 
SELECT * FROM products,category WHERE category_id = cid;

 2)  查询商品表的商品名称和价格,以及商品的分类信息 

可以通过给表起别名的方式, 方便我们的查询(有提示)

 3)  查询 格力空调是属于哪一分类下的商品 

#查询 格力空调是属于哪一分类下的商品 
SELECT p.`pname`,c.`cname` FROM products p , category c 
WHERE p.`category_id` = c.`cid` AND p.`pid` = 'p002';

<3> 显式内连接

使用 inner join ...on 这种方式, 就是显式内连接

语法格式

SELECT 字段名 FROM 左表 [INNER] JOIN 右表 ON 条件   
-- inner 可以省略

1) 查询所有商品的所有信息,和对应的分类信息

# 显式内连接查询 
SELECT * FROM products p INNER JOIN category c ON p.category_id = c.cid;

2) 查询鞋服分类下,价格大于500的商品名称和价格

# 查询鞋服分类下,价格大于500的商品名称和价格

-- 我们需要确定的几件事 -- 1.查询几张表 products & category -- 2.表的连接条件 从表.外键 = 主表的主键 -- 3.查询的条件 cname = '鞋服' and price > 500 -- 4.要查询的字段 pname price SELECT p.pname, p.price FROM products p INNER JOIN category c ON p.category_id = c.cid WHERE p.price > 500 AND cname = '鞋服';

5 多表查询 -  外连接查询 

<1> 左外连接

   左外连接 , 使用 LEFT OUTER  JOIN ,  OUTER 可以省略 

   左外连接的特点:

     以左表为基准, 匹配右边表中的数据,如果匹配的上,就展示匹配到的数据

     如果匹配不到, 左表中的数据正常展示, 右边的展示为null    

   

    1) 语法格式

SELECT 字段名 FROM 左表 LEFT [OUTER] JOIN 右表 ON 条件
-- 左外连接查询 
SELECT * FROM category c LEFT JOIN products p  ON c.`cid`= p.`category_id`;

  

  2) 左外连接, 查询每个分类下的商品个数

# 查询每个分类下的商品个数
/*
        1.连接条件: 主表.主键 = 从表.外键
        2.查询条件: 每个分类 需要分组
        3.要查询的字段: 分类名称, 分类下商品个数 
*/

-- 查询字段
SELECT c.`cname` AS '分类名称', COUNT(p.`pid`) AS '商品个数'
-- 表连接
FROM category c LEFT JOIN products p ON c.`cid` = p.`category_id` 
-- 分组
GROUP BY c.`cname`;

<2> 右外连接

    右外连接 , 使用 RIGHT OUTER  JOIN ,  OUTER 可以省略

    右外连接的特点

      以右表为基准,匹配左边表中的数据,如果能匹配到,展示匹配到的数据   

      如果匹配不到,右表中的数据正常展示, 左边展示为null

1)语法格式

SELECT 字段名 FROM 左表 RIGHT [OUTER ]JOIN 右表 ON 条件 
-- 右外连接查询 
SELECT * FROM products p RIGHT JOIN category c ON p.`category_id` = c.`cid`;

6 各种连接方式的总结

内连接: inner join , 只获取两张表中 交集部分的数据. 

左外连接: left join ,  以左表为基准 ,查询左表的所有数据, 以及与右表有交集的部分 

右外连接: right join , 以右表为基准,查询右表的所有的数据,以及与左表有交集的部分

原文地址:https://www.cnblogs.com/JasperZhao/p/15016855.html