SQL.512.游戏玩法分析II

Table: Activity

±-------------±--------+
| Column Name | Type |
±-------------±--------+
| player_id | int |
| device_id | int |
| event_date | date |
| games_played | int |
±-------------±--------+
(player_id, event_date) 是这个表的两个主键
这个表显示的是某些游戏玩家的游戏活动情况
每一行是在某天使用某个设备登出之前登录并玩多个游戏(可能为0)的玩家的记录
问题:请编写一个 SQL 查询,描述每一个玩家首次登陆的设备名称

解一:

SELECT A.player_id,A.device_id,A.event_date
FROM Activity A
GROUP BY A.player_id
HAVING A.event_date = min(A.event_date)

错误之处在于多选中了一个event_date字段,但如果不选中这个字段就没法使用HAVING语句,换成select min(event_date),但依然多一个字段。

解二:subquery

SELECT PLAYER_ID, DEVICE_ID
FROM ACTIVITY
WHERE (PLAYER_ID, EVENT_DATE) IN 
(SELECT PLAYER_ID,MIN(EVENT_DATE)
FROM ACTIVITY
GROUP BY 1);

解三:在subquery中创建一个新的排序字段,利用它进行筛选。

SELECT PLAYER_ID, DEVICE_ID
FROM (SELECT PLAYER_ID, DEVICE_ID, RANK() OVER(PARTITION BY PLAYER_ID ORDER BY EVENT_DATE) AS `RANK`
      FROM ACTIVITY) AS A
WHERE `RANK` = 1;

over不能单独使用,要和分析函数:rank(),dense_rank(),row_number()等一起使用。
其参数:over(partition by columnname1 order by columnname2)
含义:按columname1分组,根据columnname2的值进行排序。

*补一个WHERE和HAVING的区别在于:

where 子句的作用是对查询结果进行分组前,将不符合where条件的行去掉,即在分组之前过滤数据,where条件中不能包含聚组函数,使用where条件过滤出特定的行。

having 子句的作用是筛选满足条件的组,即在分组之后过滤数据,条件中经常包含聚组函数,使用having 条件过滤出特定的组,也可以使用多个分组标准进行分组。

SQL语句顺序:
SELECT [DISTINCT|DISINCTROW|ALL] select_expression,… – 查询结果
[FROM table_references – 指定查询的表
[WHERE where_definition] – where子句,查询数据的过滤条件
[GROUP BY col_name,…] – 对[匹配where子句的]查询结果进行分组
[HAVING where_definition] – 对分组后的结果进行条件限制有group by才有having)
[ORDER BY{unsigned_integer | col_name | formula} [ASC | DESC],…] – 对查询结果进行排序
[LIMIT [offset,] rows] – 对查询的显示结果进行条数限制
[PROCEDURE procedure_name] --查询存储过程返回的结果集数据
]

实际执行顺序:
1.from …
2.where …
3.group by …
4.select
5.having …
6.order by …

猜你喜欢

转载自blog.csdn.net/shinytrainee/article/details/114106302