百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

从零开始学MySQL(十九):子查询

wptr33 2024-11-17 16:43 20 浏览

学习目标

  • 能够写出子查询的SQL语句

子查询

1. 子查询的介绍

在一个 select 语句中,嵌入了另外一个 select 语句, 那么被嵌入的 select 语句称之为子查询语句,外部那个select语句则称为主查询.

主查询和子查询的关系:

  1. 子查询是嵌入到主查询中
  2. 子查询是辅助主查询的,要么充当条件,要么充当数据源
  3. 子查询是可以独立存在的语句,是一条完整的 select 语句

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),
  CONSTRAINT products_fk FOREIGN KEY (category_id) REFERENCES category (cid)
);

# 插入分类数据
INSERT INTO category(cid,cname) VALUES('c001','家电');
INSERT INTO category(cid,cname) VALUES('c002','服饰');
INSERT INTO category(cid,cname) VALUES('c003','化妆品');

# 插入商品数据
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','雷神',5000,'1','c001');
INSERT INTO products (pid, pname,price,flag,category_id) VALUES('p004','JACK JONES',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','花花公子',440,'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');

简单来讲,子查询就是在一个sql语句中嵌套另外一个SQL语句。

所以子查询中的SQL语句,必须可以单独正确的使用,可以查询出正确内容。

例1. 查询当前商品中大于平均价格的商品:

第一步:获取平均价格 (得到的是一个值)

SELECT AVG(price) FROM products;

第二步:查询所有商品

SELECT * FROM products;

第三步:将第一步结果作为第二步的查询条件,这种方式即为子查询

SELECT *
FROM
    products
WHERE
    price > (SELECT AVG(price) FROM products);

例2:获取所有商品中,平均价格大于1000的分类的全部商品:

第一步:获取平均价格大于1000的分类

SELECT
    category_id
FROM
    products
GROUP BY
    category_id
HAVING
    AVG(price) > 1000;

第二步:查询所有商品

SELECT * FROM products;

第三步:第一步查询出来的分类,作为第二步的条件进行筛选

SELECT *
FROM
    products
WHERE
        category_id IN (SELECT
                            category_id
                        FROM
                            products
                        GROUP BY
                            category_id
                        HAVING
                            AVG(price) > 1000);


4.练习

练习1:获取所有商品中价格最高的商品(最高价值商品不只有一个)

SELECT *
FROM
    products
WHERE
    price = (SELECT MAX(price) FROM products);

?

练习2:获取所有商品中,和c003类中最贵商品价格相同的其他商品

SELECT *
FROM
    products
WHERE
      price = (SELECT max(price) FROM products WHERE category_id = 'c003')
  AND category_id != 'c003';

?

练习3:获取全部商品中,价格和c002中商品价格相同的其他商品

SELECT *
FROM
    products
WHERE
      category_id != 'c002'
  AND price IN (SELECT price FROM products WHERE category_id = 'c002');


5.as关键字

在SQL查询,中可以使用as 给表或者字段起别名。

5.1给表起别名

例1:查询每个分类名称所对应的商品数量

SELECT
    c.cname,
    COUNT(*)
FROM
    products AS p
        LEFT JOIN category AS c ON c.cid = p.category_id
GROUP BY
    c.cname;

?

使用 表名 as 别名 的形式可以给表起别名,==当给表起别名后,在使用该表时只能使用别名,原名已经失效==

如果此时使用原表名将会报错,例如使用category.cname,则出现错误信息:known column 'category.cname' in 'field list'

在给表起别名时, as关键字可以省略,但是初学时不建议省略,省略后可读性较差,写法如下:

SELECT
    c.cname,
    COUNT(*)
FROM
    products p
        LEFT JOIN category c ON c.cid = p.category_id
GROUP BY
    c.cname;

5.2给字段起别名

例2:查询每个分类名称所对应的商品数量,并筛选上品总数大于3的分组

SELECT
    # 不建议将字段起中文别名
    c.cname  AS category_name,
    COUNT(*) AS total
FROM
    products p
        LEFT JOIN category c ON c.cid = p.category_id
GROUP BY
    category_name
HAVING
    total >= 3;

?

给字段定义别名时,as关键字也可以省略,写法如下:

SELECT
    # 不建议将字段起中文别名
    c.cname  category_name,
    COUNT(*) total
FROM
    products p
        LEFT JOIN category c ON c.cid = p.category_id
GROUP BY
    category_name
HAVING
    total >= 3;

3. 小结

  • 子查询是一个完整的SQL语句,子查询被嵌入到一对小括号里面
  • select语句可以作为值出现,可以作为列出现,也可以作为表出现(了解)
  • 作为值出现:一般情况都是获取聚合函数的值
  • 作为列出现:一般都是作为一个范围进行数据筛选 (in)

相关推荐

Linux高性能服务器设计

C10K和C10M计算机领域的很多技术都是需求推动的,上世纪90年代,由于互联网的飞速发展,网络服务器无法支撑快速增长的用户规模。1999年,DanKegel提出了著名的C10问题:一台服务器上同时...

独立游戏开发者常犯的十大错误

...

学C了一头雾水该咋办?

学C了一头雾水该怎么办?最简单的方法就是你再学一遍呗。俗话说熟能生巧,铁杵也能磨成针。但是一味的为学而学,这个好像没什么卵用。为什么学了还是一头雾水,重点就在这,找出为什么会这个样子?1、概念理解不深...

C++基础语法梳理:inline 内联函数!虚函数可以是内联函数吗?

上节我们分析了C++基础语法的const,static以及this指针,那么这节内容我们来看一下inline内联函数吧!inline内联函数...

C语言实战小游戏:井字棋(三子棋)大战!文内含有源码

井字棋是黑白棋的一种。井字棋是一种民间传统游戏,又叫九宫棋、圈圈叉叉、一条龙、三子旗等。将正方形对角线连起来,相对两边依次摆上三个双方棋子,只要将自己的三个棋子走成一条线,对方就算输了。但是,有很多时...

C++语言到底是不是C语言的超集之一

C与C++两个关系亲密的编程语言,它们本质上是两中语言,只是C++语言设计时要求尽可能的兼容C语言特性,因此C语言中99%以上的功能都可以使用C++完成。本文探讨那些存在于C语言中的特性,但是在C++...

在C++中,如何避免出现Bug?

C++中的主要问题之一是存在大量行为未定义或对程序员来说意外的构造。我们在使用静态分析器检查各种项目时经常会遇到这些问题。但正如我们所知,最佳做法是在编译阶段尽早检测错误。让我们来看看现代C++中的一...

ESL-通过事件控制FreeSWITCH

通过事件提供的最底层控制机制,允许我们有效地利用工具箱,适时选择使用其中的单个工具。FreeSWITCH是一个核心交换与混合矩阵,它周围有几十个模块提供各种功能特性。我们完全控制了所有的即时信息,这些...

物理老师教你学C++语言(中篇)

一、条件语句与实验判断...

C语言入门指南

当然!以下是关于C语言入门编程的基础介绍和入门建议,希望能帮你顺利起步:C语言入门指南...

C++选择结构,让程序自动进行决策

什么是选择结构?正常的程序都是从上至下顺序执行,这就是顺序结构...

C++特性使用建议

1.引用参数使用引用替代指针且所有不变的引用参数必须加上const。在C语言中,如果函数需要修改变量的值,参数必须为指针,如...

C++程序员学习Zig指南(中篇)

1.复合数据类型结构体与方法的对比C++类:...

研一自学C++啃得动吗?

研一自学C++啃得动吗?在开始前我有一些资料,是我根据网友给的问题精心整理了一份「C++的资料从专业入门到高级教程」,点个关注在评论区回复“888”之后私信回复“888”,全部无偿共享给大家!!!个人...

C++关键字介绍

下表列出了C++中的常用关键字,这些关键字不能作为变量名或其他标识符名称。1、autoC++11的auto用于表示变量的自动类型推断。即在声明变量的时候,根据变量初始值的类型自动为此变量选择匹配的...