197. Rising Temperature - Detailed Explanation
Problem Statement
You are given a table named Weather with the following columns:
- Id (int): A unique identifier for each record.
- RecordDate (date): The date of the record.
- Temperature (int): The temperature on that date.
Your task is to write a SQL query to output all records (i.e., return their Id and Temperature) where the temperature is higher than the temperature on the previous day (the record immediately preceding it by date).
Example
Consider the following sample data in the Weather table:
Id | RecordDate | Temperature |
---|---|---|
1 | 2015-01-01 | 10 |
2 | 2015-01-02 | 25 |
3 | 2015-01-03 | 20 |
4 | 2015-01-04 | 30 |
Output:
Id | Temperature |
---|---|
2 | 25 |
4 | 30 |
Explanation:
- For record with Id 2: The temperature (25) is greater than the previous day’s temperature (10 from 2015-01-01).
- For record with Id 4: The temperature (30) is greater than the previous day’s temperature (20 from 2015-01-03).
Constraints
- The table contains at least two records.
- The records have unique dates and are stored with valid dates.
Hints
-
Self-Join or Window Function:
Think about how you can compare a record’s temperature with the previous day’s temperature. One common method is to join the table with itself where the joining condition matches a record to the record from the previous day. Another modern approach is to use a window function like LAG. -
Date Comparison:
Since you are comparing consecutive days, ensure that you correctly compute the difference between dates. Most SQL dialects provide functions such as DATEDIFF that can help with this.
Approach Overview
There are two common ways to solve this problem:
1. Self-Join Approach
-
Idea:
Use a self-join to pair each record with the record from the previous day. In the join condition, check that the date difference is exactly 1 day. Then, in the WHERE clause, select only those records where the current day’s temperature is higher than that of the previous day. -
Steps:
- Alias the Weather table as w1 for the current record.
- Alias the same table as w2 for the previous day’s record.
- Use a date function (like DATEDIFF) to join the records where the difference in RecordDate is 1.
- In the WHERE clause, filter out records where w1.Temperature is greater than w2.Temperature.
-
SQL Example:
SELECT w1.Id, w1.Temperature FROM Weather w1 JOIN Weather w2 ON DATEDIFF(w1.RecordDate, w2.RecordDate) = 1 WHERE w1.Temperature > w2.Temperature;
2. Window Function (LAG) Approach
-
Idea:
Use the LAG window function to access the previous day’s temperature for each record. This method avoids an explicit self-join. -
Steps:
- Use LAG(Temperature) OVER (ORDER BY RecordDate) to retrieve the temperature of the previous day.
- In an outer query, filter the records where the current day’s temperature is greater than the value returned by LAG.
-
SQL Example:
SELECT Id, Temperature FROM ( SELECT Id, Temperature, LAG(Temperature) OVER (ORDER BY RecordDate) AS PrevTemperature FROM Weather ) AS t WHERE Temperature > PrevTemperature;
Code Implementation
Below are the two SQL solutions:
Self-Join Solution
SELECT w1.Id, w1.Temperature FROM Weather w1 JOIN Weather w2 ON DATEDIFF(w1.RecordDate, w2.RecordDate) = 1 WHERE w1.Temperature > w2.Temperature;
Window Function (LAG) Solution
SELECT Id, Temperature FROM ( SELECT Id, Temperature, LAG(Temperature) OVER (ORDER BY RecordDate) AS PrevTemperature FROM Weather ) AS t WHERE Temperature > PrevTemperature;
Step-by-Step Walkthrough and Visual Example
Example Walkthrough Using Self-Join
-
Table Data:
Assume the table Weather has the following rows:Id RecordDate Temperature 1 2015-01-01 10 2 2015-01-02 25 3 2015-01-03 20 4 2015-01-04 30 -
Self-Join Process:
- The query joins w1 and w2 where the date difference is exactly 1 day.
- For w1.Id = 2 (2015-01-02, Temperature = 25), w2 finds the record with RecordDate = 2015-01-01 (Temperature = 10). Since 25 > 10, record 2 qualifies.
- For w1.Id = 4 (2015-01-04, Temperature = 30), w2 finds the record with RecordDate = 2015-01-03 (Temperature = 20). Since 30 > 20, record 4 qualifies.
- Record 1 and record 3 do not have a previous day record that meets the criteria (or their temperatures are not rising).
-
Final Result:
The query returns records with Id 2 and Id 4.
Example Walkthrough Using Window Function
-
Inner Query:
-
The inner query computes LAG(Temperature) OVER (ORDER BY RecordDate) for each record:
Id RecordDate Temperature PrevTemperature 1 2015-01-01 10 NULL 2 2015-01-02 25 10 3 2015-01-03 20 25 4 2015-01-04 30 20
-
-
Outer Query:
- The outer query filters for rows where Temperature > PrevTemperature.
- This condition is met for:
- Id 2: 25 > 10.
- Id 4: 30 > 20.
- Record 1 is skipped because PrevTemperature is NULL, and record 3 is skipped because 20 is not greater than 25.
-
Final Result:
Again, the query returns records with Id 2 and Id 4.
Common Mistakes & Edge Cases
-
Common Mistakes:
- Forgetting to handle the first record (which does not have a previous day) when using window functions.
- Using an incorrect date difference calculation that might include non-consecutive dates.
- Overcomplicating the query by not taking advantage of SQL’s built-in functions (like DATEDIFF or LAG).
-
Edge Cases:
-
When there is only one record in the table (in which case no record qualifies).
-
When consecutive records have equal temperatures—the query should only return records where the temperature strictly rises.
-
When dates are not consecutive; ensure the join condition only considers a one-day gap.
-
Alternative Variations & Related Problems
-
Alternative Variations:
-
Falling Temperature:
Modify the query to find records where the temperature is lower than the previous day’s temperature. -
Temperature Change:
Calculate and output the difference in temperature between consecutive days. -
Continuous Rising Streak:
Find the longest streak of days where the temperature continuously rises.
-
-
Related Problems for Further Practice:
GET YOUR FREE
Coding Questions Catalog
