【MySQL】巧解客户连续递增交易

力扣题

1、题目地址

2701. 连续递增交易

2、模拟表

表:Transactions

字段名类型
transaction_idint
customer_idint
transaction_datedate
amountint
  • transaction_id 是该表的主键。
  • 每行包含有关交易的信息,包括唯一的 (customer_id, transaction_date),以及相应的 customer_id 和 amount。

3、要求

编写一个 SQL 查询,找出至少连续三天 amount 递增的客户。

并包括 customer_id 、连续交易期的起始日期和结束日期。

一个客户可以有多个连续的交易。

返回结果并按照 customer_id 升序 排列。

4、示例

输入:

Transactions 表:

transaction_idcustomer_idtransaction_dateamount
11012023-05-01100
21012023-05-02150
31012023-05-03200
41022023-05-0150
51022023-05-03100
61022023-05-04200
71052023-05-01100
81052023-05-02150
91052023-05-03200
101052023-05-04300
111052023-05-12250
121052023-05-13260
131052023-05-14270

输出:

customer_idconsecutive_startconsecutive_end
1012023-05-012023-05-03
1052023-05-012023-05-04
1052023-05-122023-05-14

解释:

  • customer_id 为 101 的客户在 2023年5月1日 至 2023年5月3日 期间进行了连续递增金额的交易。
  • customer_id 为 102 的客户没有至少连续三天的交易。
  • customer_id 为 105 的客户有两组连续交易:从 2023年5月1日 至 2023年5月4日,以及 2023年5月12日 至 2023年5月14日。
  • 结果按 customer_id 升序排序

5、代码编写

我的代码

1、将不同客户的时间正序排序,将序号排出来。
2、使用 lag 窗口函数数据往后推,往后推之后第一条数据默认值设为0保证有值,由此来判断交易金额递增。
3、将金额差值负数两种情况的数据剔除(因为存在 1,-1,1,1 的情况第二位不能剔除,我们需要把 -1,-1,1 将第一位剔除,-1,1,-1 将第一位剔除),并将序号标上。
4、重新对数据剔除后的数据进行编号,分组根据不同客户和交易时间减去序号天数,然后时间正序查询出来,这是因为原先的序号因为数据剔除的缘故不准需要再次编号。
5、我们只需要根据不同客户和交易时间减去序号天数进行分组,然后查询数量是大于等于3的数据,开始时间(因为存在两种情况 1,1,-1,1,1 和 1,-1,1,1,等下还需进行处理),结束时间直接用开始时间加数量减1天数去推算就行
6、我们只需要将时间加1天和时间加2天查询出来如果是负数,直接取时间最大值就行,将上面开始时间替换就行

with tmp as (select *, row_number() over(partition by customer_id order by transaction_date) AS rnfrom Transactions
), tmp2 as (select customer_id,transaction_date,amount - lag(amount, 1, 0) over(partition by customer_id, date_sub(transaction_date, interval rn day) order by transaction_date) AS diffAmountfrom tmp
), tmp3 as (select customer_id,transaction_date,row_number() over(partition by customer_id order by transaction_date) AS rnfrom tmpwhere not exists (select customer_id, transaction_datefrom tmp2where customer_id = tmp.customer_idand transaction_date = tmp.transaction_dateand diffAmount < 0and (not exists (select * from tmp2 where customer_id = tmp.customer_id and transaction_date in(date_add(tmp.transaction_date, interval 1 day))and diffAmount > 0 ) OR not exists(select * from tmp2 where customer_id = tmp.customer_id and transaction_date in(date_add(tmp.transaction_date, interval 1 day))and diffAmount < 0UNION ALLselect * from tmp2 where customer_id = tmp.customer_id and transaction_date in(date_add(tmp.transaction_date, interval 2 day))and diffAmount > 0)))
), tmp4 as (select customer_id,transaction_date,row_number() over(partition by customer_id, date_sub(transaction_date, interval rn day) order by transaction_date) AS rnfrom tmp3
), tmp5 as (select customer_id, transaction_date AS consecutive_start, date_add(transaction_date, interval count(*)-1 day) AS consecutive_end,count(*) AS numfrom tmp4group by customer_id, date_sub(transaction_date, interval rn day)having count(*) >= 3
) 
select tmp5.customer_id, max(if(tmp2.diffAmount<0, tmp2.transaction_date, tmp5.consecutive_start)) AS consecutive_start, tmp5.consecutive_end
from tmp5, tmp2
where tmp2.customer_id = tmp5.customer_id 
and tmp2.transaction_date in (date_add(tmp5.consecutive_start, interval 1 day) , date_add(tmp5.consecutive_start, interval 2 day) )
group by 1, 3

网友代码第一种(巧解)

1、确定其下一个日期金额 > 当前日期的行
2、利用 [当前日期-分组row_number]分连续日期的组 date_group
3、组内数据超过 2 条则为满足至少连续 3 天(因为最后一条日期关联不到下一个日期,在一开始就过滤掉了),然后给 max 日期+1

select customer_id,min(transaction_date) as consecutive_start,date_add(max(transaction_date), interval 1 day) as consecutive_end
from (select customer_id,transaction_date,date_sub(transaction_date, interval row_number() over(partition by customer_id order by transaction_date) day) as date_groupfrom (select t1.customer_id,t1.transaction_datefrom Transactions t1left join Transactions t2 on t1.customer_id = t2.customer_id and t1.transaction_date = date_sub(t2.transaction_date, interval 1 day)   where t1.amount < t2.amount) a   
) b
group by customer_id,date_group
having count(customer_id) >= 2
order by customer_id

网友代码第二种(和我的思路比较接近)

# table1用于筛选出连续日期
with table1 as (select *,datediff(transaction_date, '1970-01-01') - row_number() over (partition by customer_id order by transaction_date) flagfrom transactions
),
# table2用于筛选出table1条件下的连续递增标志
table2 as (select *,if(amount > lag(amount, 1, amount) over (partition by customer_id, flag order by transaction_date), 1, 0) isIncfrom table1
),
# table3用于筛选出table2条件下的连续日期
table3 as (select *,datediff(transaction_date, '1970-01-01') - row_number() over (partition by customer_id, flag order by transaction_date) newFlagfrom table2where isInc = 1
)
select customer_id,# 因为判断递增是比较上一条,则默认递增时第一条忽略,所以这里减一,同时下面having里为2min(transaction_date) - interval 1 day consecutive_start,max(transaction_date)                  consecutive_end
from table3
group by customer_id, newFlag
having count(*) >= 2
order by customer_id;

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/765787.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

总结: HQL语句

总结: HQL语句 Part1 数据库的操作Part2 数据表的操作1. 创建普通表2. 内外部表3. 内外部表转换 Part1 数据库的操作 查看数据库: show databases; 创建数据库: create database if not exists 数据库名 使用数据库: use 数据库名; 查看数据库详细信息: desc database 数据库名…

Echarts 利用多X轴实现未来15天天气预报

Echarts 利用多X轴实现未来15天天气预报 UI 设计图 Echarts 实现效果 代码实现 代码分解 echarts 图表上下均显示数据 通过设置 grid.top 和 grid.bottom 设置白天和夜间天气展示区域 grid: {top: 36%,bottom: 36%,left: 5%,right: 5%}, 天气图标的设置 由于 axisLabel 的…

【Linux】一文了解【进程优先级相关知识点】&【PRI / NI值】背后的修正原理(13)

前言 大家好吖&#xff0c;欢迎来到 YY 滴Linux系列 &#xff0c;热烈欢迎&#xff01; 本章主要内容面向接触过Linux的老铁 主要内容含&#xff1a; 欢迎订阅 YY滴C专栏&#xff01;更多干货持续更新&#xff01;以下是传送门&#xff01; YY的《C》专栏YY的《C11》专栏YY的《…

算法 之 排序算法

&#x1f389;欢迎大家观看AUGENSTERN_dc的文章(o゜▽゜)o☆✨✨ &#x1f389;感谢各位读者在百忙之中抽出时间来垂阅我的文章&#xff0c;我会尽我所能向的大家分享我的知识和经验&#x1f4d6; &#x1f389;希望我们在一篇篇的文章中能够共同进步&#xff01;&#xff01;&…

实验7-2-10 简易连连看(PTA)

题目&#xff1a; 本题要求实现一个简易连连看游戏模拟程序。 给定一个2N2N的方阵网格游戏盘面&#xff0c;每个格子中放置一些符号。这些符号一定是成对出现的&#xff0c;同一个符号可能不止一对。程序读入玩家给出的一对位置(x1​,y1​)、(x2​,y2​)&#xff0c;判断这两…

leetcode(Hot100)——数组篇

1、两数之和 本题使用哈希法&#xff0c;用一个哈希Map保存数组的值以及对应下标&#xff0c;代码如下&#xff1a; class Solution {public int[] twoSum(int[] nums, int target) {HashMap<Integer,Integer> map new HashMap<>();for(int i0; i<nums.length…

【网络基础】网络层基本协议介绍

目录 一、IP数据包 1.1 网络层的功能 1.2 IP数据包格式 二、ICMP协议介绍 2.1 作用 2.2 常用命令 2.2.1 Ping命令 2.2.2 tracert命令 2.3 广播域 三、ARP协议介绍 3.1 作用 3.2 原理 一、IP数据包 1.1 网络层的功能 定义了基于IP协议的逻辑地址&#xff0c;就是I…

数据结构应用——哈夫曼树

哈夫曼树 哈夫曼树的相关概念构造哈夫曼树基础算法 哈夫曼编码 哈夫曼树的相关概念 结点的权&#xff1a;有某种现实含义的数值。结点的带权路径长度&#xff1a;从树的根结点到该结点的路径长度&#xff08;经过的边数&#xff09;与该结点上权值的乘积。树的带权路径长度&am…

CAPL (Communication Access Programming Language)

CAPL (Communication Access Programming Language) 是一种专门用于模拟和测试汽车网络的脚本语言&#xff0c;特别是CAN (Controller Area Network) 和LIN (Local Interconnect Network) 网络。虽然CAPL主要用于模拟网络行为和测试网络节点&#xff0c;但它也支持一些基本的编…

XiBe希贝奶瓶好用吗?2名宝宝的宝爸深度测评分享!

几乎每个新手宝爸宝妈都会有一段时间对孩子的很多东西都是不懂的&#xff0c;一边摸索一边学习。列如关于奶瓶这个问题就困扰不少新手爸妈&#xff0c;特别是面对这么多的品牌的奶瓶完全不知道怎么选。 相信很多新手爸妈都十分担心奶瓶材质安全问题&#xff0c;所以我作为一名…

OpenCV+OpenCV-Contrib源码编译

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、OpenCV是什么&#xff1f;二、OpenCV 源码编译1.前期准备1.1 源码下载1.2 cmake安装1.3 vscode 安装1.4 git 安装1.5 mingw安装 2.源码编译2.1 打开cmake2.…

python(django)之流程接口管理后台开发

1、在models.py中加入流程接口表和单一接口表 代码如下&#xff1a; from django.db import models from product.models import Product# Create your models here.class Apitest(models.Model):apitestname models.CharField(流程接口名称, max_length64)apitester model…

Python学习笔记07

第十三章&#xff0c;面向对象 初识对象 生活中数据的组织 学校开学&#xff0c;要求学生填写自己的基础信息&#xff0c;一人发一张白纸&#xff0c;让学生自己填 我叫林军杰&#xff0c;今年31岁.来自山东省&#xff0c;我是男的&#xff0c;中国人 内容混乱 改为登记表…

详细分析Python中的enumerate()函数(附多个Demo)

目录 前言1. 基本知识2. Demo 前言 对于Python的基本函数&#xff0c;从实战中获取确切知识 1. 基本知识 enumerate() 接受一个可迭代对象作为输入&#xff0c;并返回一个枚举对象这个枚举对象包含了原始可迭代对象中的每个元素以及对应的索引它允许在循环中同时获取索引和值…

uboot - pinctrl - FPGA回片前测试阶段 - 设置GPIO引脚复用失败

问题描述 pinctrl设置引脚复用失败&#xff0c;没有调用到controller中的set_groups_function函数。 问题定位 pinctrl如何注册dm节点如何进行设备树中各个设备节点下的复用配置为什么没调用到控制器实现的set_groups_function函数 &gpio0 {status "okay";p…

Flutter插件开发与发布指南

在Flutter应用程序开发中&#xff0c;有时候我们会遇到需要使用原生功能的情况&#xff0c;这时候就需要编写Flutter插件来实现与原生平台的交互。本篇博客将介绍如何编写、发布和使用Flutter插件。 1. 准备工作 在开始之前&#xff0c;确保你已经安装好Flutter开发环境&…

JavaScript中改进的数组功能

ES6标准为数组添加了很多新功能&#xff0c;例如&#xff0c;创建数组的新方法&#xff0c;创建定型数组&#xff08;Typed Array)的能力等。 1、创建数组 在ES6以前&#xff0c;创建数组的方式主要有两种&#xff0c;一种是调用Array构造函数&#xff0c;另一种是用数组字面…

VMware Workstation Pro 17虚拟机超级详细搭建(含redis,nacos,docker)(一)

今天从零搭建一下虚拟机的环境&#xff0c;把nacos&#xff0c;redis等微服务组件还有数据库搭建到里面&#xff0c;首先看到的是我们最开始下载VMware Workstation Pro 17 之后的样子&#xff0c;总共一起应该有三部分因为篇幅太长了 下载地址 : VMware - Delivering a Digit…

Vue动态设置控制表格列表展现列

最近遇到一个需求,表格列表默认不展示某一列,当我设置后可以展示该列,也可以展示和隐藏别的列,起到可以对整个表格列展示隐藏控制的效果,如下示例,默认不展示“单位名称”这一列,在我点击设置齿轮后可以看到,有“单位名称”这一列,但未打钩: 当我勾中“单位名称”,…

第2讲-Memory(5)外存

磁盘存储器 组成 记录原理 性能指标 (磁盘存取时间的计算) 磁盘地址 工作过程 优点 ①存储容量大,位价格低; ②记录介质可重复使用; ③记录信息可长期保存而不丢失,甚至可脱机存档; ④非破坏性读出,读出时不需要再生。 缺点 存取速度慢,机械结构复杂,对工作环境…