当前位置: 首页 > news >正文

pta 题目(3)

9-10

1.2021年11月每天新用户的次日留存率

用户行为日志表tb_user_log

QQ截图20211208091258.png

(uid-用户ID, artical_id-文章ID, in_time-进入时间, out_time-离开时间, sign_in-是否签到)

QQ截图20211208091309.png

问题:统计2021年11月每天新用户的次日留存率(保留2位小数)

注:
次日留存率为当天新增的用户数中第二天又活跃了的用户数占比。
如果in_time-进入时间和out_time-离开时间跨天了,在两天里都记为该用户活跃过,结果按日期升序。

输出示例:
示例数据的输出结果如下

解释:
11.01有3个用户活跃101、102、103,均为新用户,在11.02只有101、103两个又活跃了,因此11.01的次日留存率为0.67;
11.02有104一位新用户,在11.03又活跃了,因此11.02的次日留存率为1.00;
11.03有105一位新用户,在11.04未活跃,因此11.03的次日留存率为0.00;
11.04没有新用户,不输出。

AC:

select t1.dt,round(count(t2.uid)/count(t1.uid),2) uv_left_rate
from (select uid,min(date(in_time)) dtfrom tb_user_log group by uid) as t1  -- 每天新用户表
left join (select uid , date(in_time) dtfrom tb_user_logunionselect uid , date(out_time)from tb_user_log) as t2 -- 用户活跃表
on t1.uid=t2.uid
and t1.dt=date_sub(t2.dt,INTERVAL 1 day)
where date_format(t1.dt,'%Y-%m') = '2021-11'
group by t1.dt
order by t1.dt

2.牛客每个人最近的登录日期(四)

输出样例:

请在这里给出输出样例。例如:

2020-10-12|3

2020-10-13|0

2020-10-14|1

2020-10-15|0

AC:

SELECT
a.date
,COUNT(b.user_id) AS new
FROM (
SELECT
DISTINCT date
FROM login ) AS a
LEFT JOIN (SELECTMIN(date) AS min_date,user_idFROM loginGROUP BY user_id
) AS b ON a.date = b.min_date
GROUP BY a.date
ORDER BY a.date

3.每篇文章同一时刻最大在看人数

AC:

select artical_id,max(cnt) as max_uv
from (select t2.uid,t2.artical_id,count(distinct t1.uid) cnt
from tb_user_log t1,tb_user_log t2
where t1.artical_id = t2.artical_id
and t1.in_time <=t2.out_time
and t1.out_time >= t2.out_time
and t1.artical_id <> 0
group by t2.uid,t2.artical_id
)c
group by artical_id
order by max_uv desc

4.最近的三笔订单

表:Customers

QQ_1729846479346.png


customer_id 是该表具有唯一值的列
该表包含消费者的信息


表:Orders

QQ_1729846501217.png


order_id 是该表具有唯一值的列
该表包含 id 为 customer_id 的消费者的订单信息
每一个消费者 每天一笔订单


写一个解决方案,找到每个用户的最近三笔订单。如果用户的订单少于 3 笔,则返回他的全部订单。

返回的结果按照 customer_name 升序 排列。如果有相同的排名,则按照 customer_id 升序 排列。如果排名还有相同,则按照 order_date 降序 排列。

输入:
Customers

QQ_1729846557455.png


Orders

QQ_1729846574475.png

AC:

select c.name customer_name,o.*
from Customers c inner join (
select o1.customer_id,o1.order_id,order_date
from Orders o1
where (
select count(distinct o2.order_date) from Orders o2 where o2.customer_id=o1.customer_id and o2.order_date>o1.order_date
)<3
order by customer_id,order_date desc
)o on c.customer_id=o.customer_id
order by 1,2,4 desc

5.指定日期的产品价格

产品数据表: Products

+---------------+---------+
| Column Name   | Type    |
+---------------+---------+
| product_id    | int     |
| new_price     | int     |
| change_date   | date    |
+---------------+---------+
(product_id, change_date) 是此表的主键(具有唯一值的列组合)。
这张表的每一行分别记录了 某产品 在某个日期 更改后 的新价格。

编写一个解决方案,找出在 2019-08-16 时全部产品的价格,假设所有产品在修改前的价格都是 10

任意顺序 返回结果表。

结果格式如下例所示。


示例 1:

输入:
Products 表:
+------------+-----------+-------------+
| product_id | new_price | change_date |
+------------+-----------+-------------+
| 1          | 20        | 2019-08-14  |
| 2          | 50        | 2019-08-14  |
| 1          | 30        | 2019-08-15  |
| 1          | 35        | 2019-08-16  |
| 2          | 65        | 2019-08-17  |
| 3          | 20        | 2019-08-18  |
+------------+-----------+-------------+
输出:
+------------+-------+
| product_id | price |
+------------+-------+
| 2          | 50    |
| 1          | 35    |
| 3          | 10    |
+------------+-------+

AC:

 select p1.product_id, ifnull(p2.new_price, 10) as price
from (select distinct product_idfrom products
) as p1 -- 所有的产品
left join (select product_id, new_price from productswhere (product_id, change_date) in (    -- 2019-08-16之前价格的最大值select product_id, max(change_date)from productswhere change_date <= '2019-08-16'group by product_id)
) as p2 -- 在 2019-08-16 之前有过修改的产品和最新的价格
on p1.product_id = p2.product_id

6.向公司 CEO 汇报工作的所有人

编写解决方案,找出所有直接或间接向公司 CEO 汇报工作的职工的 employee_id 。
由于公司规模较小,经理之间的间接关系 不超过 3 个经理 。
可以以 任何顺序 返回无重复项的结果。
返回结果示例如下。

AC:

select a.employee_id
from Employees a
join Employees b on a.manager_id = b.employee_id
join Employees c on b.manager_id = c.employee_id
where a.employee_id != 1 and c.manager_id = 1;

7.购买了产品A和产品B却没有购买产品C的顾客

AC:

select  c.customer_id,c.customer_name  
from Customers c left 
join Orders o on c.customer_id = o.customer_id
group by c.customer_id ,c.customer_name
having sum(if(o.product_name = 'A',1,0)) > 0 a
nd sum(if(o.product_name = 'B',1,0)) > 0 
and sum(if(o.product_name = 'C',1,0)) = 0

8.找到连续区间的开始和结束数字

AC:

select a.log_id as start_id ,min(b.log_id) as end_id from 
(select log_id from Logs where log_id-1 not in (select * from Logs)) a,
(select log_id from Logs where log_id+1 not in (select * from Logs)) b
where b.log_id>=a.log_id
group by a.log_id

9.查找成绩处于中游的学生

成绩处于中游的学生是指至少参加了一次测验, 且得分既不是最高分也不是最低分的学生。
编写解决方案,找出在 所有 测验中都处于中游的学生 (student_id, student_name)。不要返回从来没有参加过测验的学生。
返回结果表按照 student_id 排序。
返回结果格式如下。

image.png

AC:

SELECT student_id,student_name
FROM Student
WHERE student_id NOT IN (SELECT DISTINCT student_idFROM Exam AS E1LEFT JOIN(SELECT exam_id,MIN(score) AS min_score,MAX(score) AS max_scoreFROM ExamGROUP BY exam_id) AS E2 ON E2.exam_id = E1.exam_idWHERE E1.score = E2.min_scoreOR E1.score = E2.max_score)AND student_id IN (SELECT DISTINCT student_id FROM Exam);

10.两人之间的通话次数

AC:

select person1, person2, COUNT(duration) as call_count,SUM(duration) as total_duration
from ( select from_id as person1, to_id as person2, durationfrom Calls where from_id < to_id -- 找出from_id < to_id所以信息union allselect to_id as person1, from_id as person2, durationfrom Calls where to_id < from_id -- 找出from_id > to_id所以信息) t -- 用union all连接两个表即可
group by person1, person2;

11.最后一个能进入巴士的人

表: Queue

+-------------+---------+
| Column Name | Type    |
+-------------+---------+
| person_id   | int     |
| person_name | varchar |
| weight      | int     |
| turn        | int     |
+-------------+---------+
person_id 是这个表具有唯一值的列。
该表展示了所有候车乘客的信息。
表中 person_id 和 turn 列将包含从 1 到 n 的所有数字,其中 n 是表中的行数。
turn 决定了候车乘客上巴士的顺序,其中 turn=1 表示第一个上巴士,turn=n 表示最后一个上巴士。
weight 表示候车乘客的体重,以千克为单位。

有一队乘客在等着上巴士。然而,巴士有1000  千克 的重量限制,所以其中一部分乘客可能无法上巴士。

编写解决方案找出 最后一个 上巴士且不超过重量限制的乘客,并报告 person_name 。题目测试用例确保顺位第一的人可以上巴士且不会超重。

返回结果格式如下所示。

示例 1:

输入:
Queue 表
+-----------+-------------+--------+------+
| person_id | person_name | weight | turn |
+-----------+-------------+--------+------+
| 5         | Alice       | 250    | 1    |
| 4         | Bob         | 175    | 5    |
| 3         | Alex        | 350    | 2    |
| 6         | John Cena   | 400    | 3    |
| 1         | Winston     | 500    | 6    |
| 2         | Marie       | 200    | 4    |
+-----------+-------------+--------+------+
输出:
+-------------+
| person_name |
+-------------+
| John Cena   |
+-------------+
解释:
为了简化,Queue 表按 turn 列由小到大排序。
+------+----+-----------+--------+--------------+
| Turn | ID | Name      | Weight | Total Weight |
+------+----+-----------+--------+--------------+
| 1    | 5  | Alice     | 250    | 250          |
| 2    | 3  | Alex      | 350    | 600          |
| 3    | 6  | John Cena | 400    | 1000         | (最后一个上巴士)
| 4    | 2  | Marie     | 200    | 1200         | (无法上巴士)
| 5    | 4  | Bob       | 175    | ___          |
| 6    | 1  | Winston   | 500    | ___          |
+------+----+-----------+--------+--------------+

AC:

SELECT t1.person_name
FROM Queue t1
WHERE (SELECT SUM(t2.Weight)FROM Queue t2WHERE t2.Turn <= t1.Turn
) <= 1000
ORDER BY t1.Turn DESC
LIMIT 1

12.树节点

表:Tree

+-------------+------+
| Column Name | Type |
+-------------+------+
| id          | int  |
| p_id        | int  |
+-------------+------+
id 是该表中具有唯一值的列。
该表的每行包含树中节点的 id 及其父节点的 id 信息。
给定的结构总是一个有效的树。

树中的每个节点可以是以下三种类型之一:

  • **"Leaf"**:节点是叶子节点。

  • **"Root"**:节点是树的根节点。

  • **"lnner"**:节点既不是叶子节点也不是根节点。

编写一个解决方案来报告树中每个节点的类型。

任意顺序 返回结果表。

结果格式如下所示。

示例 1:

输入:
Tree table:
+----+------+
| id | p_id |
+----+------+
| 1  | null |
| 2  | 1    |
| 3  | 1    |
| 4  | 2    |
| 5  | 2    |
+----+------+
输出:
+----+-------+
| id | type  |
+----+-------+
| 1  | Root  |
| 2  | Inner |
| 3  | Leaf  |
| 4  | Leaf  |
| 5  | Leaf  |
+----+-------+
解释:
节点 1 是根节点,因为它的父节点为空,并且它有子节点 2 和 3。
节点 2 是一个内部节点,因为它有父节点 1 和子节点 4 和 5。
节点 3、4 和 5 是叶子节点,因为它们有父节点而没有子节点。

AC:

 SELECT id, 'Root' AS Type
FROM tree
WHERE p_id IS NULL
UNION
SELECT id, 'Leaf' AS Type
FROM tree
WHERE id NOT IN (SELECT DISTINCT p_idFROM treeWHERE p_id IS NOT NULL)
AND p_id IS NOT NULL
UNION
SELECT id, 'Inner' AS Type
FROM tree
WHEREid IN (SELECT DISTINCT p_idFROM treeWHERE p_id IS NOT NULL)
AND p_id IS NOT NULL

13.筛选昵称规则和试卷规则的作答记录

描述

现有用户信息表user_info(uid用户ID,nick_name昵称, achievement成就值, level等级, job职业方向, register_time注册时间):

插入.png


试卷信息表examination_info(exam_id试卷ID, tag试卷类别, difficulty试卷难度, duration考试时长, release_time发布时间):

插入.png


试卷作答记录表exam_record(uid用户ID, exam_id试卷ID, start_time开始作答时间, submit_time交卷时间, score得分):
 

插入.png


找到昵称以"牛客"+纯数字+"号"或者纯数字组成的用户对于字母c开头的试卷类别(如C,C++,c#等)的已完成的试卷ID和平均得分(四舍五入),按用户ID、平均分升序排序。由(部分)示例数据结果输出如下:

插入.png

AC:

select user_info.uid,exam_record.exam_id,
round(avg(score),0) as avg_score
from user_info
left join exam_record on exam_record.uid=user_info.uid
left join examination_info on examination_info.exam_id=exam_record.exam_id
where (tag like 'c%' or tag like 'C%')
and (nick_name rlike '^牛客[0-9]+号$' or nick_name rlike '^[0-9]+$')
and submit_time is not null
group by user_info.uid,exam_record.exam_id
order by user_info.uid,avg_score

在 MySQL 中,RLIKE 是用于进行正则表达式匹配的操作符,它与 REGEXP 功能相同,用于检查某个字段的值是否与给定的正则表达式匹配。RLIKEREGEXP 可以互换使用,但 RLIKE 更常用于表述正则表达式操作。

基本语法

 

sqlCopy Code

SELECT * FROM table_name WHERE column_name RLIKE 'pattern';
  • column_name: 要进行正则匹配的列。
  • pattern: 要匹配的正则表达式模式。

正则表达式的基础

  • .: 匹配任意单个字符(除了换行符)。
  • ^: 匹配字符串的开头。
  • $: 匹配字符串的结尾。
  • *: 匹配前一个字符零次或多次。
  • +: 匹配前一个字符一次或多次。
  • []: 匹配方括号内的任意一个字符。
  • |: 逻辑“或”运算符,匹配两个模式中的任何一个。

常见用法示例

1. 匹配以指定字符开头的字符串

假设有一个 users 表,其中有 name 列。如果你想找出所有以字母 "A" 开头的名字,可以这样查询:

 

sqlCopy Code

SELECT * FROM users WHERE name RLIKE '^A';
  • ^A 表示匹配以字母 "A" 开头的字符串。
2. 匹配包含数字的字符串

如果你想查询所有名字中包含数字的记录,可以使用以下查询:

 

sqlCopy Code

SELECT * FROM users WHERE name RLIKE '[0-9]';
  • [0-9] 是一个字符类,表示匹配任意一个数字。
3. 匹配包含多个条件的字符串

假设你想查找名字中同时包含字母 "A" 和数字的记录,可以使用:

 

sqlCopy Code

SELECT * FROM users WHERE name RLIKE 'A.*[0-9]';
  • A.*[0-9] 表示匹配一个以字母 "A" 开头,后面跟着零个或多个任意字符,并且字符串中至少有一个数字。
4. 匹配特定格式的字符串(如电子邮件)

你可以使用正则表达式来匹配类似电子邮件的字符串格式。例如:

 

sqlCopy Code

SELECT * FROM users WHERE email RLIKE '^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}$';
  • 这个正则表达式匹配常见的电子邮件格式。
5. 匹配任意长度的单词

如果你想匹配含有至少 3 个字母的单词,可以使用:

 

sqlCopy Code

SELECT * FROM users WHERE name RLIKE '\\b[a-zA-Z]{3,}\\b';
  • \\b 匹配单词边界。
  • [a-zA-Z]{3,} 匹配 3 个或更多字母。
6. 匹配多个可能的模式

你可以使用 | 操作符来匹配多个模式,例如匹配以 "A" 或 "B" 开头的记录:

 

sqlCopy Code

SELECT * FROM users WHERE name RLIKE '^(A|B)';
  • ^(A|B) 匹配以 "A" 或 "B" 开头的字符串。
7. 匹配空白字符

如果你要查找包含空白字符(如空格、制表符等)的记录,可以使用:

 

sqlCopy Code

SELECT * FROM users WHERE name RLIKE '\\s';
  • \\s 匹配任何空白字符(如空格、制表符、换行符等)。

区分大小写

  • 默认情况下,RLIKE 是区分大小写的。如果你想进行不区分大小写的匹配,可以在正则表达式模式中使用 (?i),或者使用 BINARY 来强制大小写敏感的匹配。

例如,如果要不区分大小写地查找以字母 "a" 或 "A" 开头的名字:

 

sqlCopy Code

SELECT * FROM users WHERE name RLIKE '^[aA]';

或者:

 

sqlCopy Code

SELECT * FROM users WHERE name RLIKE BINARY '^a';

注意事项

  1. 性能: 正则表达式匹配通常会比简单的 LIKE= 运算符查询慢,尤其是当数据量较大时,正则表达式的匹配会影响性能。因此,建议仅在必要时使用正则表达式。

  2. 特殊字符转义: 如果正则表达式中包含特殊字符(如 .*? 等),需要使用双反斜杠 \\ 来进行转义。

  3. 字符集支持: RLIKE 默认使用 MySQL 的字符集对正则表达式进行匹配。如果表的字符集与正则表达式的字符集不一致,可能会导致匹配失败。

总结

  • RLIKE 是 MySQL 中用于进行正则表达式匹配的操作符。
  • 可以用于匹配复杂的模式,例如数字、字母、邮箱地址等。
  • 需要注意正则表达式的效率以及特殊字符的转义问题。


http://www.mrgr.cn/news/78360.html

相关文章:

  • 【Excel/WPS】根据平均值,生成两列/多列指定范围的随机数/随机凑出两列数据
  • 机器学习模型评估指标
  • Vue中el-tree结合vuedraggable实现跨组件元素拖拽
  • Windows11环境下设置MySQL8字符集utf8mb4_unicode_ci
  • 走进 Web3 社交:打破边界,重构人际关系网络
  • 《浮岛风云》V1.0中文学习版
  • 服务器记录所有用户docker操作,监控删除容器/镜像的人
  • 自动化运维(k8s)之微服务信息自动抓取:namespaceName、deploymentName等全解析
  • ComfyUI节点安装笔记
  • 数据结构--图
  • 【NLP 3、深度学习简介】
  • 1-深度学习干货总结(持续更新)
  • Matlab Simulink HDL Coder FPGA开发初体验—计数器
  • 表征对齐在训练DiT模型中的重要性
  • 【算法 python A*算法的实现】
  • 某j,mybatis-plus,多租户,多表关联查询 ,主表不追加租户条件bug解决
  • element ui select绑定的值是对象的属性时,显示异常.
  • SAP学习
  • Android 图形系统之一:概览
  • 【Zookeeper】三,Zookeeper的安装与基本操作
  • 《Learning Three.js》学习(1)使用Three.js创建三维场景
  • ABAP OOALV模板
  • 蓝桥杯备赛笔记(一)
  • transformer学习笔记-神经网络原理
  • mini-spring源码分析
  • Leetcode(快慢指针习题思路总结,持续更新。。。)