news 2026/6/10 12:49:26

SQL查询连续登录用户方法详解

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
SQL查询连续登录用户方法详解

1. 基本数据表结构

假设有一个登录记录表:

CREATETABLElogin_log(user_idINT,login_dateDATE);

示例数据:

INSERTINTOlogin_logVALUES(1,'2024-01-01'),(1,'2024-01-02'),(1,'2024-01-03'),(1,'2024-01-05'),-- 这里断了一天(1,'2024-01-06'),(1,'2024-01-07'),(2,'2024-01-01'),(2,'2024-01-02'),(2,'2024-01-04');-- 这里断了一天

2. 方法一:使用窗口函数(推荐)

2.1 查询连续登录3天及以上的用户

WITHranked_logsAS(SELECTuser_id,login_date,ROW_NUMBER()OVER(PARTITIONBYuser_idORDERBYlogin_date)asrnFROMlogin_logGROUPBYuser_id,login_date-- 去重,防止一天多次登录),date_diffAS(SELECTuser_id,login_date,DATE_SUB(login_date,INTERVALrnDAY)asgroup_dateFROMranked_logs)SELECTuser_id,MIN(login_date)asstart_date,MAX(login_date)asend_date,COUNT(*)asconsecutive_daysFROMdate_diffGROUPBYuser_id,group_dateHAVINGCOUNT(*)>=3ORDERBYuser_id,start_date;

2.2 使用LEAD/LAG函数的简化版

WITHconsecutive_groupsAS(SELECTuser_id,login_date,LAG(login_date)OVER(PARTITIONBYuser_idORDERBYlogin_date)asprev_date,LEAD(login_date)OVER(PARTITIONBYuser_idORDERBYlogin_date)asnext_dateFROM(SELECTDISTINCTuser_id,login_dateFROMlogin_log)t)SELECTuser_id,login_dateasconsecutive_dateFROMconsecutive_groupsWHERE-- 连续3天的情况:当前日期、前1天、前2天都存在(login_date=prev_date+INTERVAL1DAYANDlogin_date=prev_date+INTERVAL2DAY)OR-- 或者检查连续区间(login_date=prev_date+INTERVAL1DAYANDlogin_date=next_date-INTERVAL1DAY)OR(login_date=next_date-INTERVAL1DAYANDlogin_date=next_date-INTERVAL2DAY);

3. 方法二:使用自连接

SELECTDISTINCTl1.user_idFROMlogin_log l1JOINlogin_log l2ONl1.user_id=l2.user_idANDl2.login_date=l1.login_date+INTERVAL1DAYJOINlogin_log l3ONl1.user_id=l3.user_idANDl3.login_date=l1.login_date+INTERVAL2DAYWHEREEXISTS(SELECT1FROMlogin_logWHEREuser_id=l1.user_idANDlogin_date=l1.login_date+INTERVAL1DAY)ANDEXISTS(SELECT1FROMlogin_logWHEREuser_id=l1.user_idANDlogin_date=l1.login_date+INTERVAL2DAY);

4. 方法三:使用递归CTE(复杂但功能强大)

WITHRECURSIVE consecutive_loginAS(-- 基础查询:每个用户的首次登录SELECTuser_id,login_date,login_dateasstart_date,1asconsecutive_daysFROM(SELECTuser_id,login_date,ROW_NUMBER()OVER(PARTITIONBYuser_idORDERBYlogin_date)asrnFROMlogin_log)tWHERErn=1UNIONALL-- 递归部分:查找连续的下一天SELECTcl.user_id,ll.login_date,CASEWHENll.login_date=cl.login_date+INTERVAL1DAYTHENcl.start_dateELSEll.login_dateENDasstart_date,CASEWHENll.login_date=cl.login_date+INTERVAL1DAYTHENcl.consecutive_days+1ELSE1ENDasconsecutive_daysFROMconsecutive_login clJOINlogin_log llONcl.user_id=ll.user_idANDll.login_date>cl.login_dateWHEREll.login_date=(SELECTMIN(login_date)FROMlogin_logWHEREuser_id=cl.user_idANDlogin_date>cl.login_date))SELECTuser_id,MAX(consecutive_days)asmax_consecutive_daysFROMconsecutive_loginGROUPBYuser_idHAVINGMAX(consecutive_days)>=3;

5. 实用查询示例

5.1 查询每个用户的最大连续登录天数

WITHranked_logsAS(SELECTuser_id,login_date,DATE_SUB(login_date,INTERVALROW_NUMBER()OVER(PARTITIONBYuser_idORDERBYlogin_date)DAY)asgroup_dateFROM(SELECTDISTINCTuser_id,login_dateFROMlogin_log)t)SELECTuser_id,MAX(consecutive_days)asmax_consecutive_daysFROM(SELECTuser_id,group_date,COUNT(*)asconsecutive_daysFROMranked_logsGROUPBYuser_id,group_date)groupsGROUPBYuser_idORDERBYmax_consecutive_daysDESC;

5.2 查询指定时间段内的连续登录

WITHranked_logsAS(SELECTuser_id,login_date,DATE_SUB(login_date,INTERVALROW_NUMBER()OVER(PARTITIONBYuser_idORDERBYlogin_date)DAY)asgroup_dateFROM(SELECTDISTINCTuser_id,login_dateFROMlogin_logWHERElogin_dateBETWEEN'2024-01-01'AND'2024-01-31')t)SELECTuser_id,COUNT(*)asconsecutive_days,MIN(login_date)asstart_date,MAX(login_date)asend_dateFROMranked_logsGROUPBYuser_id,group_dateHAVINGCOUNT(*)>=7-- 连续7天登录ORDERBYconsecutive_daysDESC;

5.3 查询连续登录中断的情况

WITHlogin_gapsAS(SELECTuser_id,login_date,LEAD(login_date)OVER(PARTITIONBYuser_idORDERBYlogin_date)asnext_date,DATEDIFF(LEAD(login_date)OVER(PARTITIONBYuser_idORDERBYlogin_date),login_date)asgap_daysFROM(SELECTDISTINCTuser_id,login_dateFROMlogin_log)t)SELECTuser_id,login_dateaslast_login_before_gap,next_dateasnext_login_after_gap,gap_days-1asmissed_daysFROMlogin_gapsWHEREgap_days>1ORDERBYuser_id,login_date;

6. 性能优化建议

  1. 创建索引
CREATEINDEXidx_user_loginONlogin_log(user_id,login_date);
  1. 分区表:如果数据量很大,按月份或用户ID范围分区

  2. 物化视图:对于频繁查询的结果可以创建物化视图

  3. 定期清理:删除历史数据,只保留最近N天的数据

7. 不同数据库的语法差异

函数/特性MySQLPostgreSQLSQL ServerOracle
日期加减DATE_ADD()+ INTERVALDATEADD()+ INTERVAL
日期差DATEDIFF()-DATEDIFF()-
行号ROW_NUMBER()ROW_NUMBER()ROW_NUMBER()ROW_NUMBER()
递归CTE支持(8.0+)支持支持支持

选择哪种方法取决于:

  • 数据量:大数据量建议使用窗口函数
  • 查询频率:频繁查询建议建立物化视图
  • 数据库版本:确保支持相关函数
  • 业务需求:是否需要实时结果还是可接受延迟
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/10 11:20:35

vLLM多进程设计:兼容性挑战与解决方案

vLLM 多进程设计:兼容性挑战与工程权衡 在大模型推理系统中,性能优化往往不只是算法或调度的比拼,更是一场与底层运行时环境的博弈。vLLM 作为当前主流的高性能推理引擎,其核心优势之一便是通过 PagedAttention 和连续批处理实现…

作者头像 李华
网站建设 2026/6/10 2:36:55

移动端适配:Android调用GPT-SoVITS生成语音方案

移动端适配:Android调用GPT-SoVITS生成语音方案 在智能语音交互日益普及的今天,用户不再满足于“能说话”的机器音,而是期待更自然、更具个性的声音体验。从有声书朗读到虚拟助手,从教育辅助到无障碍服务,个性化语音合…

作者头像 李华
网站建设 2026/6/9 20:54:47

Excalidraw日志收集方案:ELK栈整合实例

Excalidraw日志收集方案:ELK栈整合实例 在现代远程协作日益深入的今天,可视化工具早已不再是简单的“画图软件”,而是团队沟通、产品设计和系统架构讨论的核心载体。Excalidraw 作为一款开源的手绘风格白板应用,凭借其轻量、直观且…

作者头像 李华
网站建设 2026/6/10 11:21:10

清华镜像站加速TensorRT及相关工具链下载体验

清华镜像站加速TensorRT及相关工具链下载体验 在AI模型从实验室走向生产线的过程中,推理性能的优化往往比训练本身更具挑战。一个在服务器上跑出高精度的模型,若无法在实际场景中快速响应请求,其商业价值将大打折扣。尤其是在视频分析、自动驾…

作者头像 李华
网站建设 2026/6/3 10:56:37

17、GNU Make高级功能探索

GNU Make高级功能探索 1. 生成XML物料清单(BOM) 在使用标准GNU make输出时,很难回答“构建了什么以及为什么构建”的问题。可以使用一种简单的技术让GNU make创建包含物料清单(BOM)的XML文件,该BOM包含makefile构建的所有文件的名称,并以嵌套形式显示每个文件的先决条…

作者头像 李华
网站建设 2026/6/10 10:49:24

python基于Flask和Vue的电商管理系统_4nrl1242_论文

目录已开发项目效果实现截图关于我系统介绍开发技术路线核心代码参考示例本项目开发思路结论源码lw获取/同行可拿货,招校园代理 :文章底部获取博主联系方式!已开发项目效果实现截图 同行可拿货,招校园代理 python基于Flask和Vue的电商管理系统_4nrl12…

作者头像 李华