Je soupçonne qu'ils peuvent être quelque peu optimisés, mais ces requêtes devraient vous donner les résultats souhaités. Ils partagent les mêmes 3 premiers CTE qui génèrent le diff_max
valeur pour chaque data_max
. Dans la première requête, nous recherchons simplement un changement dans cette valeur (de NULL
à une valeur, ou une diminution de la valeur) afin de générer les lignes de sortie. Les 4ème et 5ème CTE de la deuxième requête sont similaires à la première requête, mais nous ajoutons un RANK
au diff_max
valeurs, afin que nous puissions JOIN
la valeur minimale (avec sa date associée) au date_diff_from
et date_diff_to
valeurs du 6ème CTE (qui est la même que ma réponse à votre autre question
).
Question 1 :
WITH cte AS (SELECT DATE(`date_time`) AS `date`,
`data`,
MAX(`data`) OVER (ORDER BY `date_time`) AS `data_max`
FROM `test`),
cte2 AS (SELECT `date`,
`data`,
`data_max`,
CASE WHEN `data` < `data_max` THEN `data` - `data_max` END AS `data_diff`
FROM cte),
cte3 AS (SELECT `date`,
MIN(`data_diff`) OVER (PARTITION BY `data_max` ORDER BY `date`) AS `diff_max`
FROM cte2),
cte4 AS (SELECT `date`, `diff_max`, LAG(`diff_max`) OVER (ORDER BY `date`) AS `old_diff_max`
FROM cte3)
SELECT `date`, `diff_max`
FROM cte4
WHERE `diff_max` < `old_diff_max` OR `old_diff_max` IS NULL AND `diff_max` IS NOT NULL
Sortie :
date diff_max
2017-01-04 -3
2017-01-09 -7
2017-01-11 -10
2017-01-13 -2
Question 2 :
WITH cte AS (SELECT DATE(`date_time`) AS `date`,
`data`,
MAX(`data`) OVER (ORDER BY `date_time`) AS `data_max`
FROM `test`),
cte2 AS (SELECT `date`,
`data`,
`data_max`,
CASE WHEN `data` < `data_max` THEN `data` - `data_max` END AS `data_diff`
FROM cte),
cte3 AS (SELECT `data_max`, `date`,
MIN(`data_diff`) OVER (PARTITION BY `data_max` ORDER BY date) AS `diff_max`
FROM cte2),
cte4 AS (SELECT `data_max`, `date`, `diff_max`,
LAG(`diff_max`) OVER (ORDER BY `date`) AS `old_diff_max`
FROM cte3),
cte5 AS (SELECT `date`, `diff_max`,
RANK() OVER (PARTITION BY `data_max` ORDER BY `diff_max`) AS `diff_rank`
FROM cte4
WHERE `diff_max` < `old_diff_max` OR `old_diff_max` IS NULL AND `diff_max` IS NOT NULL),
cte6 AS (SELECT `data_max`,
MIN(CASE WHEN `data_diff` IS NOT NULL THEN date END) AS diff_date_from,
MAX(CASE WHEN `data_diff` IS NOT NULL THEN date END) AS diff_date_to
FROM cte2
GROUP BY `data_max`
HAVING diff_date_from IS NOT NULL)
SELECT diff_date_from, diff_date_to, `date` AS diff_max_date, `diff_max`
FROM cte6
JOIN cte5 ON cte5.date BETWEEN cte6.diff_date_from AND cte6.diff_date_to
WHERE cte5.diff_rank = 1
Sortie :
diff_date_from diff_date_to diff_max_date diff_max
2017-01-04 2017-01-06 2017-01-04 -3
2017-01-09 2017-01-11 2017-01-11 -10
2017-01-13 2017-01-13 2017-01-13 -2