前言
本文主要记录牛客网真题-SQL81 牛客的课程订单分析(五),原题链接点击此处
1. 题目介绍
有很多同学在牛客购买课程来学习,购买会产生订单存到数据库里。
有一个订单信息表(order_info),简况如下:
第1行表示user_id为557336的用户在2025-10-10的时候使用了client_id为1的客户端下了C++课程的订单,但是状态为没有购买成功。
第2行表示user_id为230173543的用户在2025-10-12的时候使用了client_id为2的客户端下了Python课程的订单,状态为购买成功。
…
最后1行表示user_id为557336的用户在2025-10-26的时候使用了client_id为1的客户端下了Python课程的订单,状态为购买成功。
请你写出一个sql语句查询在2025-10-15以后,如果有一个用户下单2个以及2个以上状态为购买成功的C++课程或Java课程或Python课程,那么输出这个用户的user_id,以及满足前面条件的第一次购买成功的C++课程或Java课程或Python课程的日期first_buy_date,以及满足前面条件的第二次购买成功的C++课程或Java课程或Python课程的日期second_buy_date,以及购买成功的C++课程或Java课程或Python课程的次数cnt,并且输出结果按照user_id升序排序,以上例子查询结果如下:
解析:
id为4,6的订单满足以上条件,输出57,id为4的订单为第一次购买成功,输出first_buy_date为2025-10-23,id为6的订单为第二次购买,输出second_buy_date为2025-10-24,总共成功购买了2次;
id为5,7,8的订单满足以上条件,输出557336,id为5的订单为第一次购买成功,输出first_buy_date为2025-10-23,id为7的订单为第二次购买,输出second_buy_date为2025-10-25,总共成功购买了3次;
2. 解题思路
2.1 简便方法
1、首先计算出符合条件的所有数据
2、在第1步的基础上计算出购买成功的总条数以及根据user_id分组后再根据日期对数据进行排序。(根据题意排序要使用dense_rank()排序的方式来)
3、最后使用max()搭配case when语句即可查找到最终的结果。
2.2 复杂方法
1、首先计算出符合条件的所有数据
2、在第1步的基础上计算出购买成功的总条数以及根据user_id分组后再根据日期对数据进行排序(根据题意排序要使用dense_rank()排序的方式来),此时将表命名为o4。
3、在第2步的基础上,使用o4表再连接o5(o5内容与o4完全一样),然后挑出o4表中各user_id排序为1的数据,再挑出o5表中各user_id排序为2的数据,输出即可。
3. 解题过程
简便方法与复杂方法到第3步之前的操作和结果都完全一样,故3.1将先介绍截止到第3步之前的结果,3.2再写明简便方法的代码,3.3写明复杂方法的代码。
3.1 截至第3步前的代码
select o.*,count(user_id)over(partition by user_id)cnt,
dense_rank()over(partition by user_id order by date) ro
from (select * from order_info
where date>'2025-10-15' and product_name in('C++','Java','Python') and status='completed'
)o
3.2 简便方法
select user_id,max(case when ro=1 then date else 0 end),max(case when ro=2 then date else 0 end),cnt
from(
select o.*,count(user_id)over(partition by user_id)cnt,
dense_rank()over(partition by user_id order by date) ro
from (select * from order_info
where date>'2025-10-15' and product_name in('C++','Java','Python') and status='completed'
)o)o1
where cnt>1
group by user_id
order by user_id
3.3 复杂方法
select distinct o4.user_id,o4.date,o5.date,o4.cnt from(
select o.*,count(user_id)over(partition by user_id)cnt,
dense_rank()over(partition by user_id order by date) ro
from (select * from order_info
where date>'2025-10-15' and product_name in('C++','Java','Python') and status='completed'
)o)o4 join
(
select o.*,count(user_id)over(partition by user_id)cnt,
dense_rank()over(partition by user_id order by date) ro
from (select * from order_info
where date>'2025-10-15' and product_name in('C++','Java','Python') and status='completed'
)o)o5 on o4.user_id=o5.user_id
where o4.ro=1 and o5.ro=2 and o4.cnt>1 and o5.cnt>1
order by o4.user_id