数据库 求一个 MySQL 查询语句

kai209209 · 2018年04月15日 · 最后由 kai209209 回复于 2018年04月15日 · 6319 次阅读

各位大佬,我现在有 2 个表,一个是 orders,一个是 packages。 下面的语句是找出 orders 的 state 为 4,并且 order 对应的 packages 状态也为 4 的所有订单

select
  orders.*
from
  orders, packages
where
  orders.id = packages.order_id
and
  orders.state = 4
and
  packages.state = 4
group by
  packages.order_id
having
  count(packages.id) > 1

但是这个语句找出来的结果是 packages 包含了 state 为 4 所有订单,这些 orders 对应的所有包裹里除了 packages.state 为 4,也可能还包含了其他状态,我想找出 orders 对应的所有 packages.state 里状态只有 4 的订单,请问一下这个我怎么写这个 sql 啊? 麻烦大家了

按照你的描述你写的就是对的

select o.* from orders o left join packages p on o.id = p.order_id where o.status = 4 and p.status = 4

sum case when

对每个 order 而言,找它的 package 的数量和状态为 4 的 package 的数量相等的就行了。

SELECT o.*
FROM orders o
JOIN packages p ON p.order_id = o.id
WHERE o.state = 4
GROUP BY o.id
HAVING COUNT(p.id) = SUM(IF(p.state = 4, 1, 0))
;

为了更贴近 SQL 标准(也方便以后弃坑 MySQL),把 if 换成 case 更好:SUM(CASE WHEN p.state = 4 THEN 1 ELSE 0 END) 。算是补充炮哥言简意赅的回答吧。

谢谢两位 @hooopo @darkbaby123 其实我也想到一个比较流氓的方法的

Order.joins(:packages).where(state: 4).group(:id).having("count(packages.id) > 1 and max(packages.state) = 4 and min(packages.state) = 4")

  => SELECT `orders`.* FROM `orders` INNER JOIN `packages` ON `packages`.`order_id` = `orders`.`id` WHERE `orders`.`state` = 4 GROUP BY `orders`.`id` HAVING (count(packages.id) > 1 and max(packages.state) = 4 and min(packages.state) = 4)
需要 登录 后方可回复, 如果你还没有账号请 注册新账号