题解 | #月总刷题数和日均刷题数#
月总刷题数和日均刷题数
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
子句中,你只能使用实际的列名或表达式,不能使用由表达式计算出来的别名。
最后不要忘记排序