Maison > base de données > tutoriel mysql > Comment utiliser les fonctions de fenêtrage dans MySQL

Comment utiliser les fonctions de fenêtrage dans MySQL

WBOY
Libérer: 2023-05-30 15:10:36
avant
2837 Les gens l'ont consulté

(1) Définition de la fonction de fenêtrage

La fonction de fenêtrage est également appelée fonction OLAP (Online Analytical Processing, traitement analytique en ligne), qui est principalement utilisée pour analyser et traiter des données en temps réel. Avant MySQL version 8.0, les fonctions de fenêtrage n'étaient pas prises en charge, mais la prise en charge des fonctions de fenêtrage est fournie depuis cette version.

# 开窗函数语法 
func_name(<parameter>) 
OVER([PARTITION BY <part_by_condition>] 
[ORDER BY <order_by_list> ASC|DESC])
Copier après la connexion

Analyse de l'instruction de la fonction de fenêtre :
La fonction est divisée en deux parties, une partie est le nom de la fonction, le nombre de fonctions de fenêtre est relativement faible, il n'y a que 11 fonctions de fenêtre + fonctions d'agrégation au total (toutes les fonctions d'agrégation peuvent être utilisé comme fonction de fonctions de fenêtre). Selon la nature de la fonction, certaines doivent écrire des paramètres et d'autres non.

L'autre partie est l'instruction over. over() doit être écrite. Les paramètres à l'intérieur sont tous facultatifs et peuvent être utilisés de manière sélective en fonction des besoins :

  • Le premier paramètre est le champ partition by +, ce qui signifie basé sur Ce champ divise l'ensemble de données en plusieurs parties

  • Le deuxième paramètre est le champ Trier par +. Les données de chaque fenêtre sont classées par ordre croissant ou décroissant en fonction de ce champ

Comment utiliser les fonctions de fenêtrage dans MySQL

Fonction de fenêtrage et regroupement. fonction d'agrégation Elles sont relativement similaires et les données sont divisées en plusieurs parties en spécifiant des champs. La différence est la suivante :

  • La norme SQL permet à toutes les fonctions d'agrégation d'être utilisées comme fonctions de fenêtre, et le mot-clé OVER est utilisé pour distinguer les fonctions de fenêtre. et fonctions d'agrégation.

  • La fonction d'agrégation ne renvoie qu'une seule valeur par groupe, tandis que la fonction de fenêtrage peut renvoyer plusieurs valeurs par groupe.

Parmi ces 11 fonctions de fenêtrage, les trois fonctions de tri ROW_NUMBER(), RANK() et DENSE_RANK() sont les plus utilisées dans le travail réel. Apprenons ces trois fonctions de fenêtrage à travers un simple ensemble de données.

# 首先创建虚拟的业务员销售数据 
CREATE TABLE Sales
( 
idate date, 
iname char(2), 
sales int
); 
# 向表中插入数据 
INSERT INTO Sales VALUES 
(&#39;2021/1/1&#39;, &#39;丁一&#39;, 200), 
(&#39;2021/2/1&#39;, &#39;丁一&#39;, 180), 
(&#39;2021/2/1&#39;, &#39;李四&#39;, 100), 
(&#39;2021/3/1&#39;, &#39;李四&#39;, 150), 
(&#39;2021/2/1&#39;, &#39;刘猛&#39;, 180), 
(&#39;2021/3/1&#39;, &#39;刘猛&#39;, 150), 
(&#39;2021/1/1&#39;, &#39;王二&#39;, 200), 
(&#39;2021/2/1&#39;, &#39;王二&#39;, 180), 
(&#39;2021/3/1&#39;, &#39;王二&#39;, 300), 
(&#39;2021/1/1&#39;, &#39;张三&#39;, 300), 
(&#39;2021/2/1&#39;, &#39;张三&#39;, 280), 
(&#39;2021/3/1&#39;, &#39;张三&#39;, 280); 
# 数据查询 
SELECT * FROM Sales; 
# 查询各月中销售业绩最差的业务员
SELECT month(idate),iname,sales, 
	ROW_NUMBER() 
	OVER(PARTITION BY month(idate) 
			 ORDER BY sales) as sales_order 
FROM Sales;

SELECT * FROM 
(SELECT month(idate),iname,sales, 
	 ROW_NUMBER() 
	 OVER(PARTITION BY month(idate) 
   ORDER BY sales) as sales_order FROM Sales) as t
WHERE sales_order=1;
Copier après la connexion

Comment utiliser les fonctions de fenêtrage dans MySQL

# ROW_NUMBER()、RANK()、DENSE_RANK()的区别 
SELECT * FROM 
(SELECT month(idate) as imonth,iname,sales, 
ROW_NUMBER() 
OVER(PARTITION BY month(idate) ORDER BY sales) as row_order,
RANK() 
OVER(PARTITION BY month(idate) ORDER BY sales) as rank_order, 
DENSE_RANK() 
OVER(PARTITION BY month(idate) ORDER BY sales) as dense_order 
FROM Sales) as t;
Copier après la connexion

Comment utiliser les fonctions de fenêtrage dans MySQL

ROW_NUMBER() : tri séquentiel——1, 2, 3
RANK() : tri parallèle, ignorer les numéros de série répétés——1, 1, 3
DENSE_RANK() : Parallèle trier, sans sauter les numéros de série répétés - 1, 1, 2

(2) Scénarios d'application pratiques des fonctions de fenêtrage

Au travail ou lors d'entretiens, vous pouvez rencontrer la nécessité pour les utilisateurs de se connecter pendant des jours consécutifs ou de se connecter pendant un certain nombre de jours . Ce qui suit fournit une idée pour utiliser les fonctions de fenêtrage pour résoudre de tels problèmes.

# 首先创建虚拟的用户登录表,并插入数据 
create table user_login
( 
user_id varchar(100), 
login_time datetime
); 

insert into user_login values 
(1,&#39;2020-11-25 13:21:12&#39;), 
(1,&#39;2020-11-24 13:15:22&#39;), 
(1,&#39;2020-11-24 10:30:15&#39;), 
(1,&#39;2020-11-24 09:18:27&#39;), 
(1,&#39;2020-11-23 07:43:54&#39;), 
(1,&#39;2020-11-10 09:48:36&#39;), 
(1,&#39;2020-11-09 03:30:22&#39;), 
(1,&#39;2020-11-01 15:28:29&#39;), 
(1,&#39;2020-10-31 09:37:45&#39;), 
(2,&#39;2020-11-25 13:54:40&#39;), 
(2,&#39;2020-11-24 13:22:32&#39;), 
(2,&#39;2020-11-23 10:55:52&#39;), 
(2,&#39;2020-11-22 06:30:09&#39;), 
(2,&#39;2020-11-21 08:33:15&#39;), 
(2,&#39;2020-11-20 05:38:18&#39;), 
(2,&#39;2020-11-19 09:21:42&#39;), 
(2,&#39;2020-11-02 00:19:38&#39;), 
(2,&#39;2020-11-01 09:03:11&#39;), 
(2,&#39;2020-10-31 07:44:55&#39;), 
(2,&#39;2020-10-30 08:56:33&#39;), 
(2,&#39;2020-10-29 09:30:28&#39;); 
# 查看数据 
SELECT * FROM user_login;
Copier après la connexion

Il existe généralement trois situations lors du calcul du nombre de jours de connexion consécutifs :

  • Afficher la situation de connexion continue de chaque utilisateur

  • Afficher le nombre maximum de jours de connexion consécutifs pour chaque utilisateur

  • Afficher dans une certaine période de temps Utilisateurs qui se sont connectés en continu pendant plus de N jours

Pour la première situation : vérifiez la situation de connexion continue de chaque utilisateur
Sur la base de l'expérience réelle, nous savons que dans un certain laps de temps, les utilisateurs peuvent se connecter plusieurs fois en continu. Nous utilisons ces informations. Tous les champs doivent être affichés, de sorte que les champs affichés dans le résultat final peuvent être l'ID utilisateur, la date de première connexion, la date de fin de connexion et le nombre de jours de connexion consécutifs.

# 数据预处理:由于统计的窗口期是天数,所以可以对登录时间字段进行格式转换,将其变成日期格式然后再去重(去掉用户同一天内多次登录的情况) 
# 为方便后续代码查看,将处理结果放置新表中,一步一步操作 
create table user_login_date(
select distinct user_id, date(login_time) login_date from user_login);
# 处理后的数据如下: 
select * from user_login_date;

# 第一种情况:查看每位用户连续登陆的情况 
# 对用户登录数据进行排序 
create table user_login_date_1( 
select *,
rank() over(partition by user_id order by login_date) irank 
from user_login_date); 
#查看结果 
select * from user_login_date_1;
 
# 增加辅助列,帮助判断用户是否连续登录 
create table user_login_date_2( 
select *,
date_sub(login_date, interval irank DAY) idate  #data_sub从指定的日期减去指定的时间间隔
from user_login_date_1); 
# 查看结果 
select * from user_login_date_2; 

# 计算每位用户连续登录天数 
select user_id, 
min(login_date) as start_date, 
max(login_date) as end_date, 
count(login_date) as days 
from user_login_date_2 
group by user_id,idate;

# ===============【整合代码,解决用户连续登录问题】=================== 
select user_id, 
       min(login_date) start_date, 
       max(login_date) end_date, 
       count(login_date) days 
from (select *,date_sub(login_date, interval irank day) idate 
from (select *,rank() over(partition by user_id order by login_date) irank 
from (select distinct user_id, date(login_time) login_date from user_login) as a) as b) as c 
group by user_id,idate;
Copier après la connexion

Pour le deuxième cas : vérifiez le nombre maximum de jours de connexion consécutifs pour chaque utilisateur

# 计算每个用户最大连续登录天数 
select user_id,max(days) from 
(select user_id, 
			 min(login_date) start_date, 
			 max(login_date) end_date, 
			 count(login_date) days 
from (select *,date_sub(login_date, interval irank day) idate 
from (select *,rank() over(partition by user_id order by login_date) irank 
from (select distinct user_id, date(login_time) login_date from user_login) as a) as b) as c 
group by user_id,idate) as d 
group by user_id;
Copier après la connexion

Pour le troisième cas : vérifiez les utilisateurs qui se sont connectés pendant plus de N jours sur une certaine période

Si nous devons vérifier le nombre de jours de connexion consécutifs sur 10. Comment mettre en œuvre cela pour les utilisateurs qui se sont connectés pendant 5 jours consécutifs ou plus entre le 29 novembre et le 25 novembre ? . Cette exigence peut également être filtrée à partir des résultats de la requête dans le premier cas.

# 查看在这段时间内连续登录天数≥5天的用户 
select distinct user_id from 
(select user_id, 
		min(login_date) start_date, 
		max(login_date) end_date, 
		count(login_date) days 
from (select *,date_sub(login_date, interval irank day) idate 
from (select *,rank() over(partition by user_id order by login_date) irank 
from (select distinct user_id, date(login_time) login_date from user_login) as a) as b) as c 
group by user_id,idate 
having days>=5
) as d;
Copier après la connexion

Cette façon d'écrire peut donner des résultats, mais elle est un peu gênante pour ce problème. Voici une méthode simple : faites référence à une nouvelle fonction de fenêtre statique lead()

select *, 
lead(login_date,4) over(partition by user_id order by login_date) as idate5 
from user_login_date;
Copier après la connexion

La fonction lead a trois paramètres. le premier paramètre est la colonne spécifiée (la date de connexion est utilisée ici), le deuxième paramètre est la valeur de plusieurs lignes après la ligne actuelle, ici 4, qui est la date de la cinquième connexion, et le troisième paramètre est s'il est renvoyé Null les valeurs peuvent être remplacées par des valeurs spécifiées. Le troisième paramètre n'est pas utilisé ici. Dans la clause over, les fenêtres sont regroupées par user_id et les données de chaque fenêtre sont classées par ordre croissant par date de connexion.

Utilisez la cinquième date de connexion - login_date+1. Si elle est égale à 5, cela signifie que vous êtes connecté pendant cinq jours consécutifs. Si vous obtenez une valeur nulle ou supérieure à 5, cela signifie que vous n'êtes pas connecté. pendant cinq jours consécutifs. Le code et les résultats sont les suivants :

# 计算第5次登录日期与当天的差值 
select *,datediff(idate5,login_date)+1 days 
from (select *,lead(login_date,4) over(partition by user_id order by login_date) idate5
from user_login_date) as a; 
# 找出相差天数为5的记录 
select distinct user_id 
from (select *,datediff(idate5,login_date)+1 as days 
from (select *,lead(login_date,4) over(partition by user_id order by login_date) idate5 
from user_logrin_date) as a)as b 
where days = 5;
Copier après la connexion

[Exercice 】Questions d'entretien d'analyse des données de la plateforme de retrait Meituan - SQL
La table de données de transaction existante user_goods_table est la suivante :

Comment utiliser les fonctions de fenêtrage dans MySQL

Maintenant, le patron veut savoir la répartition des préférences de catégorie à emporter achetées par chaque utilisateur, et découvrez la catégorie à emporter la plus achetée par chaque utilisateur.

# 分析题目:要求输出字段为用户名user_name,该用户购买最多的外卖品类goods_kind 
# 解题思路:这是一个分组排序的问题,可以考虑窗口函数 
# 第一步:使用窗口函数row_number(),对每个用户购买的外卖品类进行分组统计与排名
select user_name,goods_kind,count(goods_kind),
rank() over (partition by user_name order by count(goods_kind) desc) as irank
from user_goods_table
group by user_name,goods_kind;

# 第二步:筛选出每个用户排名第一的外卖品类
select user_id,goods_kind from 
(select user_name,goods_kind,count(goods_kind),
rank() over (partition by user_name order by count(goods_kind) desc) as irank
from user_goods_table
group by user_name,goods_kind) as a 
where irank=1
Copier après la connexion

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Étiquettes associées:
source:yisu.com
Déclaration de ce site Web
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn
Tutoriels populaires
Plus>
Derniers téléchargements
Plus>
effets Web
Code source du site Web
Matériel du site Web
Modèle frontal