【MySQL】MySQL版本8+ 窗口函数 Lead 的两种使用

力扣题

1、题目地址

1709. 访问日期之间最大的空档期

2、模拟表

表:UserVisits

Column NameType
user_idint
visit_datedate
  • 该表没有主键,它可能有重复的行
  • 该表包含用户访问某特定零售商的日期日志。

3、要求

  • 假设今天的日期是 ‘2021-1-1’ 。
  • 编写解决方案,对于每个 user_id ,求出每次访问及其下一个访问(若该次访问是最后一次,则为今天)之间最大的空档期天数 window
  • 返回结果表,按用户编号 user_id 排序。

4、示例

输入:

UserVisits 表:

user_idvisit_date
12020-11-28
12020-10-20
12020-12-3
22020-10-5
22020-12-9
32020-11-11

输出:

user_idbiggest_window
139
265
351

解释:

对于第一个用户,问题中的空档期在以下日期之间:

  • 2020-10-20 至 2020-11-28 ,共计 39 天。
  • 2020-11-28 至 2020-12-3 ,共计 5 天。
  • 2020-12-3 至 2021-1-1 ,共计 29 天。

由此得出,最大的空档期为 39 天。
对于第二个用户,问题中的空档期在以下日期之间:

  • 2020-10-5 至 2020-12-9 ,共计 65 天。
  • 2020-12-9 至 2021-1-1 ,共计 23 天。

由此得出,最大的空档期为 65 天。
对于第三个用户,问题中的唯一空档期在 2020-11-11 至 2021-1-1 之间,共计 51 天。

5、代码编写

Lead (列名) 语法

SELECT user_id, MAX(biggest_window) AS biggest_window
FROM (
    SELECT *, DATEDIFF(
    	IFNULL(
    		Lead(visit_date) over (partition by user_id order by visit_date), 
    		'2021-1-1'
    	), visit_date
    ) AS biggest_window
    FROM UserVisits
) AS one
GROUP BY user_id

如果加 IFNULL

SELECT *, DATEDIFF(
	IFNULL(
		Lead(visit_date) over (partition by user_id order by visit_date), 
		'2021-1-1'
	), visit_date
) AS biggest_window
FROM UserVisits
| user_id | visit_date | biggest_window |
| ------- | ---------- | -------------- |
| 1       | 2020-10-20 | 39             |
| 1       | 2020-11-28 | 5              |
| 1       | 2020-12-03 | 29             |
| 2       | 2020-10-05 | 65             |
| 2       | 2020-12-09 | 23             |
| 3       | 2020-11-11 | 51             |

如果不加 IFNULL,还需要特别处理

SELECT *, DATEDIFF(
    Lead(visit_date) over (partition by user_id order by visit_date),  
    visit_date
) AS biggest_window
FROM UserVisits
| user_id | visit_date | biggest_window |
| ------- | ---------- | -------------- |
| 1       | 2020-10-20 | 39             |
| 1       | 2020-11-28 | 5              |
| 1       | 2020-12-03 | null           |
| 2       | 2020-10-05 | 65             |
| 2       | 2020-12-09 | null           |
| 3       | 2020-11-11 | null           |

Lead (列名, 偏移量, 超出记录窗口时的默认值) 语法

SELECT user_id, MAX(biggest_window) AS biggest_window
FROM (
    SELECT *, DATEDIFF(
    	Lead(visit_date, 1, '2021-01-01') over (partition by user_id order by visit_date), 
    	visit_date
    ) AS biggest_window
    FROM UserVisits
) AS one
GROUP BY user_id

以前文章

【MySQL】窗口函数 Lead 和 Lag 的运用(MySQL版本8+)以及 时间差函数 TIMESTAMPDIFF 的运用

  • 9
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
MySQL 中,可以使用自定义函数(User-Defined Function,简称 UDF)来实现自定义加法函数。下面是一个示例: 1. 首先,创建一个用于存储 UDF 的库: ```sql CREATE DATABASE my_udf; USE my_udf; ``` 2. 然后,编写 UDF 的 C 代码。这里使用 MySQL 提供的 `libmysqlclient` 库来操作 MySQL 数据库,代码如下: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <mysql.h> #ifdef _WIN32 #define EXPORT __declspec(dllexport) #else #define EXPORT #endif my_bool my_addition_init(UDF_INIT *initid, UDF_ARGS *args, char *message); void my_addition_deinit(UDF_INIT *initid); long long my_addition(UDF_INIT *initid, UDF_ARGS *args, char *is_null, char *error); my_bool my_addition_init(UDF_INIT *initid, UDF_ARGS *args, char *message) { if (args->arg_count != 2 || args->arg_type[0] != INT_RESULT || args->arg_type[1] != INT_RESULT) { strcpy(message, "my_addition() requires two integer arguments"); return 1; } return 0; } void my_addition_deinit(UDF_INIT *initid) { // do nothing } long long my_addition(UDF_INIT *initid, UDF_ARGS *args, char *is_null, char *error) { long long a = *((long long *) args->args[0]); long long b = *((long long *) args->args[1]); return a + b; } ``` 3. 编译代码并将生成的动态链接库复制到 MySQL 的插件目录中: ```sh gcc -shared -o my_addition.so -I/usr/include/mysql my_addition.c -lmysqlclient cp my_addition.so /usr/lib/mysql/plugin/ ``` 4. 在 MySQL 中加载 UDF: ```sql CREATE FUNCTION my_addition RETURNS INTEGER SONAME 'my_addition.so'; ``` 5. 最后,在 MySQL 中调用自定义加法函数: ```sql SELECT my_addition(8, 5); -- 输出 13 ``` 注意,以上示例仅供参考,实际使用时需要根据自己的需求进行修改和调试。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值