做题笔记:SQL Sever 方式做牛客SQL的题目--SQL156

2023-12-09 16:01

本文主要是介绍做题笔记:SQL Sever 方式做牛客SQL的题目--SQL156,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

----SQL156 各个视频的平均完播率
问题:计算2021年里有播放记录的每个视频的完播率(结果保留三位小数),并按完播率降序排序
注:视频完播率是指完成播放次数占总播放次数的比例。
简单起见,结束观看时间与开始播放时间的差>=视频时长时,视为完成播放。
输出顺序:video_id | avg_comp_play_rate

表的创建及数据的插入:

DROP TABLE IF EXISTS tb_user_video_log, tb_video_info;
CREATE TABLE tb_user_video_log
(id         INT PRIMARY KEY identity,-- '自增ID',uid        INT NOT NULL,-- '用户ID',video_id   INT NOT NULL,-- '视频ID',start_time datetime,-- '开始观看时间',end_time   datetime,-- '结束观看时间',if_follow  TINYINT,-- '是否关注',if_like    TINYINT,-- '是否点赞',if_retweet TINYINT,-- '是否转发',comment_id INT,--'评论ID'
);CREATE TABLE tb_video_info
(id           INT PRIMARY KEY identity,-- '自增ID',video_id     INT UNIQUE  NOT NULL,-- '视频ID',author       INT         NOT NULL,-- '创作者ID',tag          VARCHAR(16) NOT NULL,-- '类别标签',duration     INT         NOT NULL,-- '视频时长(秒数)',release_time datetime    NOT NULL,-- '发布时间'
);INSERT INTO tb_user_video_log(uid, video_id, start_time, end_time, if_follow, if_like, if_retweet, comment_id)
VALUES (101, 2001, '2021-10-01 10:00:00', '2021-10-01 10:00:30', 0, 1, 1, null),(102, 2001, '2021-10-01 10:00:00', '2021-10-01 10:00:24', 0, 0, 1, null),(103, 2001, '2021-10-01 11:00:00', '2021-10-01 11:00:34', 0, 1, 0, 1732526),(101, 2002, '2021-09-01 10:00:00', '2021-09-01 10:00:42', 1, 0, 1, null),(102, 2002, '2021-10-01 11:00:00', '2021-10-01 11:00:30', 1, 0, 1, null),(103, 2002, '2021-10-01 10:59:05', '2021-10-01 11:00:05', 1, 0, 1, null),(101, 2003, '2020-09-01 10:00:00', '2020-09-01 10:01:42', 1, 0, 1, null),(102, 2003, '2021-09-01 10:00:00', '2021-09-01 10:00:42', 1, 0, 1, null);INSERT INTO tb_video_info(video_id, author, tag, duration, release_time)
VALUES (2001, 901, '影视', 30, '2021-01-01 7:00:00'),(2002, 901, '美食', 60, '2021-01-01 7:00:00'),(2003, 902, '旅游', 90, '2021-01-01 7:00:00');

解题思路:
① 所查信息涉及两表,所以需要多表查询 - join
② 视频每次播放的实际时间 = 结束时间 - 开始时间

datediff(second,start_time,end_time)		--时间差,以秒为单位

③ 以视频分组,比较实际播放时间>= 视频时长(秒数) 的并统计个数 / 视频播放总次数
④ 筛选2021年的数据

查询如下:

 select video_id,Convert(decimal(18,3),count(iif(watch_time >= duration,1,null))* 1.0 / count(video_id)) avg_comp_play_ratefrom (select tu.video_id,year(start_time) as year_v,datediff(second,start_time,end_time)  as watch_time,tv.durationfrom tb_user_video_log tujoin tb_video_info tvon tu.video_id = tv.video_id)twhere year_v = '2021'group by video_id

上述查询用了一层嵌套,有利于在写查询时候理清思路,可以优化简洁代码如下:

 select tu.video_id,Convert(decimal(18,3),count(iif(datediff(second,start_time,end_time) >= duration,1,null))* 1.0 / count(tu.video_id)) avg_comp_play_ratefrom tb_user_video_log tujoin tb_video_info tvon tu.video_id = tv.video_idwhere year(start_time) = '2021'group by tu.video_id

做题总结:
① 需要注意count(iif(watch_time >= duration,1,null))的使用,在开始null处写的0是错误的,count统计返回的元素个数1或0都会被统计,所以导致最后完播率都是100%,如果返回是null则不会被count统计,可筛选出符合要求的数据。如果想反回0,可以用sum函数替换count函数:sum(iif(watch_time >= duration,1,0))
DATEDIFF 函数
DATEDIFF函数用于计算两个日期之间的差异,可以以天、小时、分钟、秒等不同的单位来返回结果。
DATEDIFF的基本语法如下:

DATEDIFF(datepart, startdate, enddate)

在次语法中:
datepart 表示需要计算的时间单位,可以是以下值之一:
“year”(年);“quarter”(季度);“month”(月)
“day”(天);“hour”(小时);“minute”(分钟)
“second”(秒)
startdate 表示起始日期;enddate 表示结束日期。
Convert 函数

Convert(decimal(18,3), x * 1.0 / y)

Convert 函数将x/y的结果转化为decimal(18,3)数据类型:decimal(18,3)是一种数据类型,用于存储具有总共18位数值的数字,其中包括3位小数,即保留三位小数

这篇关于做题笔记:SQL Sever 方式做牛客SQL的题目--SQL156的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/474284

相关文章

AJAX请求上传下载进度监控实现方式

《AJAX请求上传下载进度监控实现方式》在日常Web开发中,AJAX(AsynchronousJavaScriptandXML)被广泛用于异步请求数据,而无需刷新整个页面,:本文主要介绍AJAX请... 目录1. 前言2. 基于XMLHttpRequest的进度监控2.1 基础版文件上传监控2.2 增强版多

Docker镜像修改hosts及dockerfile修改hosts文件的实现方式

《Docker镜像修改hosts及dockerfile修改hosts文件的实现方式》:本文主要介绍Docker镜像修改hosts及dockerfile修改hosts文件的实现方式,具有很好的参考价... 目录docker镜像修改hosts及dockerfile修改hosts文件准备 dockerfile 文

Linux中的计划任务(crontab)使用方式

《Linux中的计划任务(crontab)使用方式》:本文主要介绍Linux中的计划任务(crontab)使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、前言1、linux的起源与发展2、什么是计划任务(crontab)二、crontab基础1、cro

Ubuntu中远程连接Mysql数据库的详细图文教程

《Ubuntu中远程连接Mysql数据库的详细图文教程》Ubuntu是一个以桌面应用为主的Linux发行版操作系统,这篇文章主要为大家详细介绍了Ubuntu中远程连接Mysql数据库的详细图文教程,有... 目录1、版本2、检查有没有mysql2.1 查询是否安装了Mysql包2.2 查看Mysql版本2.

基于SpringBoot+Mybatis实现Mysql分表

《基于SpringBoot+Mybatis实现Mysql分表》这篇文章主要为大家详细介绍了基于SpringBoot+Mybatis实现Mysql分表的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可... 目录基本思路定义注解创建ThreadLocal创建拦截器业务处理基本思路1.根据创建时间字段按年进

Win11安装PostgreSQL数据库的两种方式详细步骤

《Win11安装PostgreSQL数据库的两种方式详细步骤》PostgreSQL是备受业界青睐的关系型数据库,尤其是在地理空间和移动领域,:本文主要介绍Win11安装PostgreSQL数据库的... 目录一、exe文件安装 (推荐)下载安装包1. 选择操作系统2. 跳转到EDB(PostgreSQL 的

Python3.6连接MySQL的详细步骤

《Python3.6连接MySQL的详细步骤》在现代Web开发和数据处理中,Python与数据库的交互是必不可少的一部分,MySQL作为最流行的开源关系型数据库管理系统之一,与Python的结合可以实... 目录环境准备安装python 3.6安装mysql安装pymysql库连接到MySQL建立连接执行S

Java枚举类实现Key-Value映射的多种实现方式

《Java枚举类实现Key-Value映射的多种实现方式》在Java开发中,枚举(Enum)是一种特殊的类,本文将详细介绍Java枚举类实现key-value映射的多种方式,有需要的小伙伴可以根据需要... 目录前言一、基础实现方式1.1 为枚举添加属性和构造方法二、http://www.cppcns.co

MySQL双主搭建+keepalived高可用的实现

《MySQL双主搭建+keepalived高可用的实现》本文主要介绍了MySQL双主搭建+keepalived高可用的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录一、测试环境准备二、主从搭建1.创建复制用户2.创建复制关系3.开启复制,确认复制是否成功4.同

MyBatis 动态 SQL 优化之标签的实战与技巧(常见用法)

《MyBatis动态SQL优化之标签的实战与技巧(常见用法)》本文通过详细的示例和实际应用场景,介绍了如何有效利用这些标签来优化MyBatis配置,提升开发效率,确保SQL的高效执行和安全性,感... 目录动态SQL详解一、动态SQL的核心概念1.1 什么是动态SQL?1.2 动态SQL的优点1.3 动态S