题解 | #月总刷题数和日均刷题数#

月总刷题数和日均刷题数

https://www.nowcoder.com/practice/f6b4770f453d4163acc419e3d19e6746

SELECT submit_month, month_q_cnt, ROUND(month_q_cnt/day_month, 3) avg_day_q_cnt
FROM (
    SELECT DATE_FORMAT(submit_time, '%Y%m') submit_month,
    COUNT(*) month_q_cnt,
    CASE WHEN MONTH(submit_time)=9 THEN 30 ELSE 31 END day_month
    FROM practice_record
    WHERE submit_time IS NOT NULL AND YEAR(submit_time) = 2021
    GROUP BY submit_month,day_month
) a
UNION 
SELECT CONCAT(DATE_FORMAT(submit_time, '%Y'), '汇总') submit_month,
    COUNT(*) month_q_cnt,
    ROUND(COUNT(*)/31,3) avg_day_q_cnt
FROM practice_record
WHERE submit_time IS NOT NULL AND YEAR(submit_time) = 2021
GROUP BY submit_month
ORDER BY submit_month

题解:

任务:找到21年每月的月活数、平均日活数

思路:观察示例发现最后有个汇总行,因此想到UNION,它和加all的区别就在于这个会自动去重。

同时月份需要格式化,使用DATE_FORMAT函数;月活数就直接使用聚合函数;日平均活跃数需要计算字段,而且每个月的日数不一样,所以采用CASE WHEN语句;

第一个使用子查询是因为分组语句的原因,MySQL的only_full_group_by模式。在这个模式下,SELECT 语句的所有列都必须是聚合的或包含在 GROUP BY 子句中,然而ROUND(month_q_cnt/day_month, 3)是计算表达式,不能用于分组。所以使用子查询的方式避免了该类错误,最后要记得给子查询一个别名。

第二个可以看到第一个字段的值是有数字和汉字,所以需要拼接,使用concat函数。 这里的ROUND(COUNT(*)/31,3) avg_day_q_cnt不报错是因为有聚合函数,并且里面也没有不能聚合的字段。同时在 GROUP BY 子句中,你只能使用实际的列名或表达式,不能使用由表达式计算出来的别名。

最后不要忘记排序

全部评论

相关推荐

牛客刘北:如果暑期实习是27届的话,你要晚一年才会毕业,企业为什么会等你呢?要搞清时间逻辑呀!27届现在实习只能是在暑假实习,这是日常实习,不是暑期实习。所以多去投日常实习吧,暑期实习肯定不会要你的
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务