条件窗口函数

时间:2020-07-22 16:05:22

标签: mysql sql

我有一个如下的销售表:

store_id    cust_id   txn_id   txn_date   amt       industry
200         1         1        20180101   21.01     1000   
200         2         2        20200102   20.01     1000
200         2         3        20200103   19        1000
200         3         4        20180103   19        1000
200         4         5        20200103   21.01     1000
300         2         6        20200104   1.39      2000
300         1         7        20200105   12.24     2000
300         1         8        20200105   25.02     2000
400         2         9        20180106   103.1     1000
400         2         10       20200107   21.3      1000

以下是生成此样本表的代码:

CREATE TABLE sales(
    store_id INT,
    cust_id INT,
    txn_id INT,
    txn_date bigint,
    amt float,
    industry INT);

INSERT INTO sales VALUES(200,1,1,20180101,21.01,1000);
INSERT INTO sales VALUES(200,2,2,20200102,20.01,1000);
INSERT INTO sales VALUES(200,2,3,20200103,19.00,1000);
INSERT INTO sales VALUES(200,3,4,20180103,19.00,1000);
INSERT INTO sales VALUES(200,4,5,20200103,21.01,1000);
INSERT INTO sales VALUES(300,2,6,20200104,1.39,2000);
INSERT INTO sales VALUES(300,1,7,20200105,12.24,2000);
INSERT INTO sales VALUES(300,1,8,20200105,25.02,2000);
INSERT INTO sales VALUES(400,2,9,20180106,103.1,1000);
INSERT INTO sales VALUES(400,2,10,20200107,21.3,1000);

我想做的是创建一个新表results来回答这个问题:自2020年1月3日以来,我的VIP客户中有i%的人仅在我的商店购物; ii)在我的商店和同一行业的其他商店; iii)仅在同一行业的其他商店?将VIP客户定义为自2019年以来至少在指定商店购物过一次的人。

这是目标输出表:

store    industry   pct_my_store_only   pct_both   pct_other_stores_only
200      1000         0.5               0.5         0.0
300      2000         0.5               0.5         0.0
400      1000         0.0               1.0         0.0

我正在尝试使用窗口函数来完成此任务。这是我到目前为止的内容:

CREATE TABLE results as
    SELECT s.store_id, s.industry,
    COUNT(DISTINCT (CASE WHEN s.txn_date>=20200103 THEN s.cust_id END)) * 1.0 / sum(count(DISTINCT (CASE WHEN s.txn_date>=20200103 THEN s.cust_id END))) OVER (PARTITION BY s.industry) AS pct_my_store_only
    ...AS pct_both
    ...AS pct_other_stores_only
    FROM sales s
    WHERE sales.txn_date>=20190101 
    GROUP BY s.store_id, s.industry;

以上似乎不正确;我该如何纠正?

1 个答案:

答案 0 :(得分:2)

为每个客户将不同的store_id和行业加入到串联的不同的store_id和行业中,然后使用窗口函数avg()和函数find_in_set()来确定某个客户是否从中购物了多少客户每个商店:

with 
  stores as (
    select distinct store_id, industry
    from sales
    where txn_date >= 20190103
  ),
  customers as (
    select cust_id, 
           group_concat(distinct store_id) stores,
           group_concat(distinct industry) industries 
    from sales
    where txn_date >= 20190103
    group by cust_id
 ),
  cte as (
  select *,
    avg(concat(s.store_id) = concat(c.stores)) over (partition by s.store_id, s.industry) pct_my_store_only,
    avg(find_in_set(s.store_id, c.stores) = 0) over (partition by s.industry) pct_other_stores_only
    from stores s inner join customers c 
    on find_in_set(s.industry, c.industries) and find_in_set(s.store_id, c.stores)
  )  
select distinct store_id, industry, 
       pct_my_store_only,
       1 - pct_my_store_only - pct_other_stores_only pct_both,   
       pct_other_stores_only
from cte       
order by store_id, industry

请参见demo
结果:

> store_id | industry | pct_my_store_only | pct_both | pct_other_stores_only
> -------: | -------: | ----------------: | -------: | --------------------:
>      200 |     1000 |            0.5000 |   0.5000 |                0.0000
>      300 |     2000 |            0.5000 |   0.5000 |                0.0000
>      400 |     1000 |            0.0000 |   1.0000 |                0.0000