SQL Server 基于每月活动的客户生命周期状态分析

68de4m5k  于 2022-11-28  发布在  其他
关注(0)|答案(1)|浏览(145)

您好,我的公司希望更好地跟踪有多少用户在我们的平台上活跃。我们使用Microsoft SQL Server 2019作为数据库,连接到Azure Data Studio。
下面是来自我们数据库的两个表DDL:
1.日历表
| 色谱柱|数据类型|详细信息|
| - -|- -|- -|
| 日历日期|日期不为空|基准日期(年-月-日)|
| 日历年|整数非空|2010年、2011年等|
| 日历月份编号|整数非空|第1至12页|
| 日历_月份_名称|变量字符(100)|一月、二月等|
| 日历_月_日|整数非空|1-31页|
| 日历星期几|整数非空|第1至7页|
| 日历日名称|整数非空|星期一、星期二等|
| 日历年月|整数不为空,|2010年11月、2010年12月、2011年1月等|
1.收入分析
| 立柱|数据类型|详细信息|
| - -|- -|- -|
| 活动日期|日期不为空|下注日期|
| 成员标识|整数非空|唯一玩家标识符|
| 游戏ID| SMALLINT非空|唯一游戏标识符|
| 工资金额|真实的不为空|在游戏中下注的总金额|
| 工资数|整数非空|在游戏中下注的次数|
| 获胜金额|真实的不为空|游戏赢得的总金额|
| 活动年月|整数非空|年|
| 银行类型标识|小整数默认值0不为空,|0=真实的钱,1=奖金|
以下两个表格的屏幕截图:
日历表

收入分析表

长话短说,“活跃”意味着会员在当月至少进行了一次真实的钱投注。
每个月,成员都有一个特定的活动周期类型。此状态将每月根据其上个月和本月的活动进行更改。状态如下:
| 新产品|他们第一次下真实的钱赌注|
| - -|- -|
| 保留|在上一个日历月和当前日历月有效|
| 未保留|在前一个日历月有效,但在当前日历月无效|
| 重新激活|在上一个日历月不活动,但在当前日历月活动|
| 搭接|在上一个日历月或当前日历月不活动|
我们希望首先获得具有以下列的视图:
成员标识|日历_年_月|成员生命周期状态|已用_月数
此外,该视图应该显示每个成员每月一行,从他们第一次下真实的钱赌注的月份开始。该视图应该给予他们该月的生命周期状态,如果成员已经失效,它应该显示他们自上次活动以来的月数滚动计数。
到目前为止,我已经提出了下面的CTE给予我一个基础的看法。但我不确定的未保留和重新激活列。任何人的想法?

with all_activities as (
select a.member_id, activity_date, calendar_month_number as month_activity, calendar_year as year_activity, 
datepart(month,CURRENT_TIMESTAMP) as current_month, datepart(year,CURRENT_TIMESTAMP) as current_year,
datepart(month,CONVERT(DATE, DATEADD(DAY,-DAY(GETDATE()),GETDATE()))) as previous_month, datepart(year,CONVERT(DATE, DATEADD(DAY,-DAY(GETDATE()),GETDATE()))) as year_last_month,
a.NUMBER_OF_WAGERS, (case when datepart(month,CURRENT_TIMESTAMP) = calendar_month_number and datepart(year,CURRENT_TIMESTAMP) = calendar_year then 'active' else 'inactive' end) as status,
case when (case when datepart(month,CURRENT_TIMESTAMP) = calendar_month_number and datepart(year,CURRENT_TIMESTAMP) = calendar_year then 'active' else 'inactive' end) = 'active' and number_of_wagers = 1 then 'New' 
when (LAG((case when datepart(month,CURRENT_TIMESTAMP) = calendar_month_number and datepart(year,CURRENT_TIMESTAMP) = calendar_year then 'active' else 'inactive' end) ,1,0) OVER(PARTITION BY member_id ORDER BY calendar_month_number desc) = 'active' and calendar_month_number = datepart(month,CONVERT(DATE, DATEADD(DAY,-DAY(GETDATE()),GETDATE())))) then 'Retained' 
when (calendar_month_number = datepart(month,CURRENT_TIMESTAMP) and year_activity = datepart(year,CURRENT_TIMESTAMP) and calendar_month_number = datepart(month,CONVERT(DATE, DATEADD(DAY,-DAY(GETDATE()),GETDATE())))) then 'Unretained' 
from [dbo].[REVENUE_ANALYSIS] a
join CALENDAR b on a.ACTIVITY_DATE= b.CALENDAR_DATE
)
select * from all_activities
jutyujz0

jutyujz01#

这是关于客户生命周期状态分析,它需要几件事:
1.客户获得日期(最好将其存储起来,因为有些客户可能会追溯到几年或几十年前)。对于此问题,我们假设revenue_analysis包含我们需要的所有内容,并计算user acquisition month

  1. lapsedchurned的比较:一个churned的客户通常被定义为一段时间没有活动,对于这个问题,我们没有定义,因此,一个用户将永远被报告为lapsed
    1.对于生命周期状态计算,我们将收集以下信息(member_id、calendar_month、acquisition_month、activity_month、prior_activity_month),以便计算最终结果。
with cte_new_user_monthly as (
select member_id,
       min(activity_year_month) as acquisition_month
  from revenue_analysis
 group by 1),
cte_user_monthly as (
select u.member_id,
       u.acquisition_month,
       m.yyyymm as calendar_month
  from cte_new_user_monthly u,
       calendar_month m
 where u.acquisition_month <= m.yyyymm),
cte_user_activity_monthly as (
select f.member_id,
       f.activity_year_month as activity_month
  from revenue_analysis f
 group by 1,2),
cte_user_lifecycle as (
select u.member_id,
       u.calendar_month,
       u.acquisition_month,
       m.activity_month
  from cte_user_monthly u
  left
  join cte_user_activity_monthly m
    on u.member_id = m.member_id
   and u.calendar_month = m.activity_month),
cte_user_status as (
select member_id,
       calendar_month,
       acquisition_month,
       activity_month,
       lag(activity_month,1) over (partition by member_id order by calendar_month) as prior_activity_month
  from cte_user_lifecycle),
user_status_monthly as (
select member_id,
       calendar_month,
       activity_month,
       case
          when calendar_month = acquisition_month then 'NEW'
          when prior_activity_month is not null and activity_month is not null then 'RETAINED'
          when prior_activity_month is not null and activity_month is null then 'UNRETAINED'
          when prior_activity_month is null and activity_month is not null then 'REACTIVATED'
          when prior_activity_month is null and activity_month is null then 'LAPSED'
          else null
       end as user_status
  from cte_user_status)
select member_id,
       calendar_month,
       activity_month,
       user_status,
       row_number() over (partition by member_id, user_status order by calendar_month) as months
  from user_status_monthly
 order by 1,2;

结果(包括activity_month以便于理解):

member_id|calendar_month|activity_month|user_status|months|
---------+--------------+--------------+-----------+------+
     1001|        201701|        201701|NEW        |     1|
     1001|        201702|              |UNRETAINED |     1|
     1001|        201703|              |LAPSED     |     1|
     1001|        201704|              |LAPSED     |     2|
     1001|        201705|        201705|REACTIVATED|     1|
     1001|        201706|        201706|RETAINED   |     1|
     1001|        201707|              |UNRETAINED |     2|
     1001|        201708|              |LAPSED     |     3|
     1001|        201709|        201709|REACTIVATED|     2|
     1001|        201710|              |UNRETAINED |     3|
     1001|        201711|              |LAPSED     |     4|
     1001|        201712|        201712|REACTIVATED|     3|
     1002|        201703|        201703|NEW        |     1|
     1002|        201704|              |UNRETAINED |     1|
     1002|        201705|              |LAPSED     |     1|
     1002|        201706|              |LAPSED     |     2|
     1002|        201707|              |LAPSED     |     3|
     1002|        201708|              |LAPSED     |     4|
     1002|        201709|              |LAPSED     |     5|
     1002|        201710|              |LAPSED     |     6|
     1002|        201711|              |LAPSED     |     7|
     1002|        201712|              |LAPSED     |     8|

编辑:
1.在MySQL中测试的代码,因为我没有注意到'mysql'标记被删除。
1.代码中的calendar_month可以从calendar维度导出。

相关问题