集合运算
表的加减法
集合在数据库领域表示记录的集合. 具体来说,表、视图和查询的执行结果都是记录的集合, 其中的元素为表或者查询结果中的每一行。
在标准 SQL 中, 分别对检索结果使用 UNION
, INTERSECT,
EXCEPT
来将检索结果进行并,交和差运算, 像UNION
,INTERSECT
, EXCEPT
这种用来进行集合运算的运算符称为集合运算符。
以下的文氏图展示了几种集合的基本运算。
表的加法–UNION
eg:
SELECT product_id, product_name
FROM product
UNION
SELECT product_id, product_name
FROM product2;
UNION 等集合运算符通常都会除去重复的记录。这种去重不仅会去掉两个结果集相互重复的, 还会去掉一个结果集中的重复行
练习题:
假设连锁店想要增加毛利率超过 50%或者售价低于 800 的货物的存货量, 请使用 UNION 对分别满足上述两个条件的商品的查询结果求并集。
结果应该类似于:
我的解答:
SELECT product_id, product_name, product_type, sale_price, pruchase_price
FROM product
WHERE sale_price - purchase_price > purchase_price*0.5
UNION
SELECT product_id, product_name, product_type, sale_price, pruchase_price
FROM product
WHERE sale_price < 800;
不用UNION :
SELECT product_id,product_name,product_type
,sale_price,purchase_price
FROM product
WHERE sale_price < 800
OR sale_price > 1.5 * purchase_price;
对于同一个表的两个不同的筛选结果集, 使用 UNION 对两个结果集取并集, 和把两个子查询的筛选条件用 OR 谓词连接, 会得到相同的结果, 但倘若要将两个不同的表中的结果合并在一起, 就不得不使用 UNION 了。
练习题 :
分别使用 UNION 或者 OR 谓词,找出毛利率不足 30%或毛利率未知的商品。
我的解答:
-- UNION
SELECT*
FROM product
WHERE sale_price < purchase_price*1.3
UNION
SELECT*
FROM product
WHERE sale_price / purchase_price IS NULL;
-- OR
SELECT*
FROM product
WHERE sale_price < purchase_price*1.3
OR sale_price / purchase_price IS NULL;
包含重复行的集合运算 UNION ALL
在 UNION 的结果中保留重复行的语法其实非常简单,只需要在 UNION 后面添加 ALL 关键字就可以了。
练习题:
商店决定对product表中利润低于50% 或者 售价低于1000的商品提价, 请使用UNION ALL 语句将分别满足上述两个条件的结果取并集. 查询结果类似下表:
我的解答:
SELECT*
FROM product
WHERE sale_price < purchase_price*1.5
UNION ALL
SELECT*
FROM product
WHERE purchase_price<1000;
隐式数据类型转换
数据类型不完全相同, 也会通过隐式类型转换来将两个类型不同的列放在一列里显示, 例如字符串和数值类型:
SELECT product_id, product_name, '1'
FROM Product
UNION
SELECT product_id, product_name,sale_price
FROM Product2;
需要注意的是 hive中进行join关联时,关联列要避免使用隐式数据类型转换,否则容易导致数据倾斜
练习题:
使用 SYSDATE()函数可以返回当前日期时间, 是一个日期时间类型的数据, 试测试该数据类型和数值,字符串等类型的兼容性。
例如, 以下代码可以正确执行, 说明时间日期类型和字符串,数值以及缺失值均能兼容。
SELECT SYSDATE(), SYSDATE(), SYSDATE()
UNION
SELECT 'chars', 123, null
上述代码的查询结果:
MySQL 8.0 不支持交运算INTERSECT
需要用 inner join 来求得交集:
SELECT p1.product_id, p1.product_name
FROM Product p1
INNER JOIN Product2 p2
ON p1.product_id=p2.product_id
差集,补集与表的减法
当使用一个集合A减去另一个集合B的时候,对于只存在于集合B而不存在于集合A的元素, 采取直接忽略的策略,因此集合A和B做减法只是将集合A中也同时属于集合B的元素减掉。
MySQL 8.0 还不支持 EXCEPT 运算
不过, 借助第六章学过的NOT IN 谓词, 我们同样可以实现表的减法。
练习题:
找出只存在于Product表但不存在于Product2表的商品。
我的解答:
SELECT*
FROM product
WHERE product_id NOT IN (SELECT product_id FROM product2);
EXCEPT 与 NOT 谓词
通过上述练习题的MySQL解法, 我们发现, 使用 NOT IN 谓词, 基本上可以实现和SQL标准语法中的EXCEPT运算相同的效果。
练习题:
使用NOT谓词进行集合的减法运算, 求出Product表中, 售价高于2000,但利润低于30%的商品, 结果应该如下表所示。
我的解答:
SELECT *
FROM product
WHERE sale_price> 2000 AND
produce_id NOT IN (SELECT product_id
FROM product
WHERE sale_price >= purchase_price*1.3;
NTERSECT 与 AND 谓词
对于同一个表的两个查询结果而言, 他们的交INTERSECT实际上可以等价地将两个查询的检索条件用AND谓词连接来实现。
练习题:
使用AND谓词查找product表中利润率高于50%,并且售价低于1500的商品,查询结果如下所示。
参考答案
SELECT *
FROM product
WHERE sale_price <1500
AND sale_price > purchase_price*1.5)
对称差
两个集合A,B的对称差是指那些仅属于A或仅属于B的元素构成的集合. 两个集合的对称差等于 A-B并上B-A, 因此实践中可以用这个思路来求对称差。
练习题:
使用Product表和Product2表的对称差来查询哪些商品只在其中一张表, 结果类似于:
提示: 使用 NOT IN 实现两个表的差集.
我的解答:
SELECT *
FROM product
WHERE product_id NOT IN (SELECT product_id
FROM product2)
UNION
SELECT *
FROM product2
WHERE product_id NOT IN (SELECT product_id
FROM product);
借助并集和差集迂回实现交集运算 INTERSECT
通过观察集合运算的文氏图, 我们发现, 两个集合的交可以看作是两个集合的并去掉两个集合的对称差。
连结(JOIN)
UNION和INTERSECT 等集合运算, 这些集合运算的特征就是以行方向为单位进行操作.
注: 连结**更适合从多张表获取信息。
连结(JOIN)就是使用某种关联条件(一般是使用相等判断谓词"="), 将其他表中的列添加过来, 进行“添加列”的集合运算.
内连结(INNER JOIN)
-- 语法格式
FROM <tb_1> INNER JOIN <tb_2> ON <condition(s)>
-- 其中 INNER 关键词表示使用了内连结
对比一下product和shopProduct两张表, 可以发现, 商品编号列是一个公共列, 因此很自然的事情就是用这个商品编号列来作为连接的“桥梁”,将Product和ShopProduct这两张表连接起来。
SELECT SP.shop_id,SP.shop_name,SP.product_id,P.product_name,P.product_type,P.sale_price,SP.quantity
FROM ShopProduct AS SP
INNER JOIN Product AS P
ON SP.product_id = P.product_id;
关于内连结,需要注意以下三点:
要点一: 进行连结时需要在 FROM 子句中使用多张表.
之前的 FROM 子句中只有一张表, 而这次我们同时使用了 ShopProduct 和 Product 两张表,使用关键字 INNER JOIN 就可以将两张表连结在一起了:
FROM ShopProduct AS SP INNER JOIN Product AS P
要点二:必须使用 ON 子句来指定连结条件.
在进行内连结时 ON 子句是必不可少的。
ON 子句是专门用来指定连结条件的, 我们在上述查询的 ON 之后指定两张表连结所使用的列以及比较条件, 基本上, 它能起到与 WHERE 相同的筛选作用。
要点三: SELECT 子句中的列最好按照 表名.列名 的格式来使用。
当两张表的列除了用于关联的列之外, 没有名称相同的列的时候, 也可以不写表名, 但表名使得我们能够在今后的任何时间阅读查询代码的时候, 都能马上看出每一列来自于哪张表, 能够节省我们很多时间。但是, 如果两张表有其他名称相同的列, 则必须使用上述格式来选择列名, 否则查询语句会报错。
结合 WHERE 子句使用内连结
如果需要在使用内连结的时候同时使用 WHERE 子句对检索结果进行筛选, 则需要把 WHERE 子句写在 ON 子句的后边。
增加 WHERE 子句的方式有好几种:
-- 第一种:上述查询作为子查询, 用括号封装起来, 然后在外层查询增加筛选条件。
SELECT*
FROM (
SELECT SP.shop_id,SP.shop_name,SP.product_id,P.product_name,P.product_type,P.sale_price,SP.quantity
FROM ShopProduct AS SP
INNER JOIN Product AS P
ON SP.product_id = P.product_id) AS STEP1
WHERE shop_name = '东京'
AND product_type = '衣服';
-- 第二种:标准写法
SELECT SP.shop_id,SP.shop_name,SP.product_id,P.product_name,P.product_type,P.sale_price,SP.quantity
FROM ShopProduct AS SP
INNER JOIN Product AS P
ON SP.product_id = P.product_id
WHERE shop_name = '东京'
AND product_type = '衣服' ;
-- 执行顺序:FROM 子句->WHERE 子句->SELECT 子句
-- 第三种:可以将 WHERE 子句中的条件直接添加在 ON 子句中
SELECT SP.shop_id,SP.shop_name,SP.product_id,P.product_name,P.product_type,P.sale_price,SP.quantity
FROM ShopProduct AS SP
INNER JOIN Product AS P
ON (SP.product_id = P.product_id
AND shop_name = '东京'
AND product_type = '衣服' );
-- 第四种:先分别在两个表使用 WHERE 进行筛选,然后把上述两个子查询连结起来。
SELECT SP.shop_id,SP.shop_name,SP.product_id,P.product_name,P.product_type,P.sale_price,SP.quantity
FROM (-- 子查询 1:从 ShopProduct 表筛选出东京商店的信息
SELECT *
FROM ShopProduct
WHERE shop_name = '东京') AS SP
INNER JOIN -- 子查询 2:从 Product 表筛选出衣服类商品的信息
(SELECT *
FROM Product
WHERE product_type = '衣服') AS P
ON SP.product_id = P.product_id;
练习题:
找出每个商店里的衣服类商品的名称及价格等信息. 希望得到如下结果:
我的解答:
-- 1--不使用子查询
SELECT SP.shop_id, SP.shop_name, P.product_id, P.product_name, P.product_type, P.purchase_price
FROM ShopProduct AS SP
INNER JOIN Product AS P
ON SP.product_id = P.product.id
WHERE P.product_type = '衣服';
-- 2--使用子查询
SELECT SP.shop_id, SP.shop_name, P.product_id, P.product_name, P.product_type, P.purchase_price
FROM ShopProduct AS SP
INNER JOIN (
SELECT product_id, product_name, product_type, purchase_price
FROM Product
WHERE product_type = '衣服'
)AS P
ON SP.product_id = P.product.id;
练习题:
分别使用连结两个子查询和不使用子查询的方式, 找出东京商店里, 售价低于 2000 的商品信息,希望得到如下结果。
我的解答:
-- 不使用子查询
SELECT SP.shop_id, SP.shop_name, SP.product_id, SP.quantity, P.product_name, P.product_type, P.sale_price
FROM ShopProduct AS SP
INNER JOIN Product AS P
ON SP.product_id = P.product.id
WHERE P.sale_price < 2000
AND SP.shop_id = '000A';
-- 使用子查询
SELECTSP.shop_id, SP.shop_name, SP.product_id, SP.quantity, P.product_name, P.product_type, P.sale_price
FROM (SELECT shop_id, shop_name, product_id, quantity
FROM ShopProduct
WHERE shop_id = '000A')AS SP
INNER JOIN (
SELECT product_id, product_name, product_type, sale_price
FROM Product
WHERE sale_price < 2000)AS P
ON SP.product_id = P.product.id;
结合 GROUP BY 子句使用内连结
结合 GROUP BY 子句使用内连结, 需要根据分组列位于哪个表区别对待。最简单的情形, 是在内连结之前就使用 GROUP BY 子句.
练习题:
每个商店中, 售价最高的商品的售价分别是多少?
我的解答:
SELECT SP.shop_id, SP.shop_name, MAX(P.sale_price) AS max_price
FROM shopproduct AS SP
INNER JOIN product AS P
ON SP.product_id = P.product_id
GROUP BY SP.shop_id,SP.shop_name
自连结(SELF JOIN)
之前的内连结, 连结的都是不一样的两个表。但实际上一张表也可以与自身作连结, 这种连接称之为自连结. 需要注意, 自连结并不是区分于内连结和外连结的第三种连结, 自连结可以是外连结也可以是内连结, 它是不同于内连结外连结的另一个连结的分类方法。
内连结与关联子查询
找出每个商品种类当中售价高于该类商品的平均售价的商品.当时我们是使用关联子查询来实现的。使用内连结同样可以解决这个问题:
- 首先, 使用 GROUP BY 按商品类别分类计算每类商品的平均价格。
- 接下来, 将上述查询与表 Product 按照 product_type (商品种类)进行内连结。
- 最后, 增加 WHERE 子句, 找出那些售价高于该类商品平均价格的商品.完整的代码如下:
SELECT P1.product_id
,P1.product_name
,P1.product_type
,P1.sale_price
,P2.avg_price
FROM Product AS P1
INNER JOIN
(SELECT product_type,AVG(sale_price) AS avg_price
FROM Product
GROUP BY product_type) AS P2
ON P1.product_type = P2.product_type
WHERE P1.sale_price > P2.avg_price;
自然连结(NATURAL JOIN)
自然连结并不是区别于内连结和外连结的第三种连结, 它其实是内连结的一种特例–当两个表进行自然连结时, 会按照两个表中都包含的列名来进行等值内连结, 此时无需使用 ON 来指定连接条件。
SELECT * FROM shopproduct NATURAL JOIN Product
上述查询得到的结果, 会把两个表的公共列(这里是 product_id, 可以有多个公共列)放在第一列, 然后按照两个表的顺序和表中列的顺序, 将两个表中的其他列都罗列出来。
练习题:
试写出与上述自然连结等价的内连结。
-- 参考答案
SELECT SP.product_id,SP.shop_id,SP.shop_name,SP.quantity
,P.product_name,P.product_type,P.sale_price
,P.purchase_price,P.regist_date
FROM shopproduct AS SP
INNER JOIN Product AS P
ON SP.product_id = P.product_id
使用自然连结还可以求出两张表或子查询的公共部分, 例如求表 Product 和表 Product2 中的公共部分, 也可以用自然连结来实现:
SELECT * FROM Product NATURAL JOIN Product2
这个结果和书上给的结果并不一致, 少了运动 T 恤, 这是由于运动 T 恤的 regist_date 字段为空, 在进行自然连结时, 两个缺失值用等号进行比较, 结果不为真. 而连结只会返回对连结条件返回为真的那些行。
如果我们将查询语句进行修改:
SELECT *
FROM (SELECT product_id, product_name
FROM Product ) AS A
NATURAL JOIN
(SELECT product_id, product_name
FROM Product2) AS B;
那就可以得到正确的结果了:
使用连结求交集
练习题: 使用内连结求 Product 表和 Product2 表的交集。
SELECT P1.*
FROM Product AS P1
INNER JOIN Product2 AS P2
ON (P1.product_id = P2.product_id
AND P1.product_name = P2.product_name
AND P1.product_type = P2.product_type
AND P1.sale_price = P2.sale_price
AND P1.regist_date = P2.regist_date)
得到如下结果
注意上述结果和 P230 的结果并不一致–少了 product_id='0003’这一行, 观察源表数据可发现, 少的这行数据的 regist_date 为缺失值, 回忆第六章讲到的 IS NULL 谓词, 我们得知, 这是由于缺失值是不能用等号进行比较导致的。
如果我们仅仅用 product_id 来进行连结:
SELECT P1.*
FROM Product AS P1
INNER JOIN Product2 AS P2
ON P1.product_id = P2.product_id
查询结果:
这次就一致了。
外连结(OUTER JOIN)
内连结会丢弃两张表中不满足 ON 条件的行,和内连结相对的就是外连结. 外连结会根据外连结的种类有选择地保留无法匹配到的行。
按照保留的行位于哪张表,外连结有三种形式: 左连结, 右连结和全外连结。
左连结会保存左表中无法按照 ON 子句匹配到的行, 此时对应右表的行均为缺失值; 右连结则会保存右表中无法按照 ON 子句匹配到的行, 此时对应左表的行均为缺失值; 而全外连结则会同时保存两个表中无法按照 ON子句匹配到的行, 相应的另一张表中的行用缺失值填充。
三种外连结的对应语法分别为:
-- 左连结
FROM <tb_1> LEFT OUTER JOIN <tb_2> ON <condition(s)>
-- 右连结
FROM <tb_1> RIGHT OUTER JOIN <tb_2> ON <condition(s)>
-- 全外连结
FROM <tb_1> FULL OUTER JOIN <tb_2> ON <condition(s)>
左连结与右连结
由于连结时可以交换左表和右表的位置, 因此左连结和右连结并没有本质区别.
练习题: 统计每种商品分别在哪些商店有售, 需要包括那些在每个商店都没货的商品。
使用左连结
SELECT SP.shop_id, SP.shop_name, SP.product_id, P.product_name, P.sale_price
FROM ShopProduct AS SP
LEFT OUTER JOIN
Product AS P
ON SP.product_id = P.product_id
ORDER BY product_id;
● 外连结要点 1: 选取出单张表中全部的信息
由于内连结只能选取出同时存在于两张表中的数据,因此只在 Product 表中存在的 2 种商品并没有出现在结果之中.相反,对于外连结来说,只要数据存在于某一张表当中,就能够读取出来.使用外连结能够得到固定行数的结果.外连结名称的由来也跟 NULL 有关,即“结果中包含原表中不存在(在原表之外)的信息”.相反,只包含表内信息的连结也就被称为内连结了。
● 外连结要点 2:使用 LEFT、RIGHT 来指定主表.
外连结还有一点非常重要,那就是要把哪张表作为主表.最终的结果中会包含主表内所有的数据.指定主表的关键字是 LEFT 和 RIGHT.顾名思义,使用 LEFT 时 FROM 子句中写在左侧的表是主表,使用 RIGHT 时右侧的表是主表.使用哪一个都可以.通常使用 LEFT 的情况会多一些,但也并没有非使用这个不可的理由,使用 RIGHT 也没有问题。
结合 WHERE 子句使用左连结
练习题:
使用外连结从ShopProduct表和Product表中找出那些在某个商店库存少于50的商品及对应的商店.希望得到如下结果。
注意高压锅和圆珠笔两种商品在所有商店都无货, 所以也应该包括在内。
我的解答:
SELECT P.product_id, P.product_name, P.sale_price, SP.shop_id, SP.shop_name, SP.quantity
FROM
(SELECT*
FROM ShopProduct
WHERE quantity < 50) AS SP
RIGHT OUTER JOIN
Product AS P
ON SP.product_id = P.product_id
ORDER BY product_id;
在 MySQL 中实现全外连结
MySQL8.0 目前还不支持全外连结, 不过我们可以对左连结和右连结的结果进行 UNION 来实现全外连结。
多表连结
多表进行内连结
SELECT SP.shop_id
,SP.shop_name
,SP.product_id
,P.product_name
,P.sale_price
,IP.inventory_quantity
FROM ShopProduct AS SP
INNER JOIN Product AS P
ON SP.product_id = P.product_id
INNER JOIN InventoryProduct AS IP
ON SP.product_id = IP.product_id
WHERE IP.inventory_id = 'P001';
多表进行外连结
SELECT P.product_id
,P.product_name
,P.sale_price
,SP.shop_id
,SP.shop_name
,IP.inventory_quantity
FROM Product AS P
LEFT OUTER JOIN ShopProduct AS SP
ON SP.product_id = P.product_id
LEFT OUTER JOIN InventoryProduct AS IP
ON SP.product_id = IP.product_id
ON 子句进阶–非等值连结
除了使用相等判断的等值连结, 也可以使用比较运算符来进行连接. 实际上, 包括比较运算符(<,<=,>,>=, BETWEEN)和谓词运算(LIKE, IN, NOT 等等)在内的所有的逻辑运算都可以放在 ON 子句内作为连结条件。
非等值自左连结(SELF JOIN)
练习题:
希望对 Product 表中的商品按照售价赋予排名. 一个从集合论出发,使用自左连结的思路是, 对每一种商品,找出售价不低于它的所有商品, 然后对售价不低于它的商品使用 COUNT 函数计数. 例如, 对于价格最高的商品,
SELECT product_id, product_name, sale_price, COUNT(p2_id) AS my_rank
FROM (--使用自左连结对每种商品找出价格不低于它的商品
SELECT P1.product_id
,P1.product_name
,P1.sale_price
,P2.product_id AS P2_id
,P2.product_name AS P2_name
,P2.sale_price AS P2_price
FROM Product AS P1
LEFT OUTER JOIN Product AS P2
ON P1.sale_price <= P2.sale_price
) AS X
GROUP BY product_id, product_name, sale_price
ORDER BY my_rank;
练习题:
请按照商品的售价从低到高,对售价进行累计求和
首先, 按照题意, 对每种商品使用自左连结, 找出比该商品售价价格更低或相等的商品
SELECT product_id, product_name, sale_price
,SUM(P2_price) AS cum_price
FROM
(SELECT P1.product_id, P1.product_name, P1.sale_price
,P2.product_id AS P2_id
,P2.product_name AS P2_name
,P2.sale_price AS P2_price
FROM Product AS P1
LEFT OUTER JOIN Product AS P2
ON ((P1.sale_price > P2.sale_price)
OR (P1.sale_price = P2.sale_price
AND P1.product_id<=P2.product_id))
ORDER BY P1.sale_price,P1.product_id) AS X
GROUP BY product_id, product_name, sale_price
ORDER BY sale_price,cum_price;
交叉连结—— CROSS JOIN(笛卡尔积)
交叉连结的语法有如下几种形式:
-- 1.使用关键字 CROSS JOIN 显式地进行交叉连结
SELECT SP.shop_id
,SP.shop_name
,SP.product_id
,P.product_name
,P.sale_price
FROM ShopProduct AS SP
CROSS JOIN Product AS P;
--2.使用逗号分隔两个表,并省略 ON 子句
SELECT SP.shop_id
,SP.shop_name
,SP.product_id
,P.product_name
,P.sale_price
FROM ShopProduct AS SP , Product AS P;
-- 交叉连结是对两张表中的全部记录进行交叉组合,因此结果中的记录数通常是两张表中行数的乘积.
交叉连结没有应用到实际业务之中的原因有两个.一是其结果没有实用价值,二是由于其结果行数太多,需要花费大量的运算时间和高性能设备的支持。
练习题
4.1
找出 product 和 product2 中售价高于 500 的商品的基本信息。
我的解答:
SELECT*
FROM product
WHERE sale_price > 500
UNION ALL
SELECT*
FROM product2
WHERE sale_price > 500
4.2
借助对称差的实现方式, 求product和product2的交集。
我的解答:
SELECT *
FROM product
WHERE product_id NOT IN (SELECT product_id FROM product2)
UNION
SELECT *
FROM product2
WHERE product_id NOT IN (SELECT product_id FROM product);
SELECT *
FROM (SELECT *
FROM product
UNION
SELECT *
FROM product2) AS u
WHERE product_id NOT IN (
SELECT product_id
FROM product
WHERE product_id NOT IN (SELECT product_id FROM product2)
UNION
SELECT product_id
FROM product2
WHERE product_id NOT IN (SELECT product_id FROM product)
);
4.3
每类商品中售价最高的商品都在哪些商店有售 ?
我的解答:
SELECT SP.shop_id, SP.shop_name, SP.product_id, P.product_name, P.product_type, P.sale_price, MP.maxp AS '该类商品的最大售价'
FROM product AS P
INNER JOIN shop_product AS SP
ON SP.product_id = P.product_id
INNER JOIN (
SELECT product_type, MAX(sale_price) AS maxp
FROM product
GROUP BY product_type
) AS MP
ON MP.product_type = P.product_type AND P.sale_price = MP.maxp;
4.4
分别使用内连结和关联子查询每一类商品中售价最高的商品。
我的解答:
-- 内连结
SELECT P.product_id, P.product_name, P.product_type, P.sale_price
FROM product AS P
INNER JOINx
(SELECT product_type, MAX(sale_price) AS maxp
FROM product
GROUP BY product_type) AS MP
ON MP.product_type = P.product_type AND P.sale_price = MP.maxp;
-- 关联子查询
SELECT P.product_id, P.product_name, P.product_type, P.sale_price
FROM product AS P
WHERE sale_price = (SELECT MAX(sale_price)
FROM product AS P1
WHERE P.product_type = P1.product_type
GROUP BY product_type);
4.5
用关联子查询实现:在 product 表中,取出 product_id, product_name, sale_price, 并按照商品的售价从低到高进行排序、对售价进行累计求和。
我的解答:
SELECT P.product_id, P.product_name, P.sale_price,
(SELECT SUM(sale_price) FROM product AS P1
WHERE P.sale_price > P1.sale_price
OR (P.sale_price = P1.sale_price AND P.product_id >= P1.product_id)
) AS '累计求和'
FROM product AS P
ORDER BY sale_price;
okok本章结束~