Filtering on aggregated data
Key Takeaways
- In SQL we can filter on values in the original, non-aggregated data as well as values generated from an aggregate function applied over values of another column.
- To filter on values from aggregate functions grouped by values of another column, however, we need to use slightly different code.
Introduction
If you have ever written any piece of code to analyze data you probably encountered a situation where the output on the screen did not match what you wanted. If you are like me, you also may have said (or yelled) to the computer, ”Why can’t you just do what I want you to do?!” followed by a moment of realization that the code you wrote was not completely correct for completing the intended task at hand. For all the great things they can do, computers do not infer meaning and intent from people well. As such, computers require very deliberate instructions, written in code, to produce the desired result.
data:image/s3,"s3://crabby-images/2bb19/2bb1992546ae3e9c31acffa401eb6381fed226cb" alt=""
SQL is no different, and one area that is prone to a disconnect between intention and output involves filtering on aggregated data. In a previous post entitled ”Translating Basic Excel to SQL” I covered some fundamental operations in SQL, including WHERE statements, which are used to filter rows based on some condition (the materials used in that prior article are also used in this article and can be downloaded here). For example, the media below shows the execution of a WHERE statement in DbVisualizer.
Execution of a WHERE statement
↓
data:image/s3,"s3://crabby-images/ba57d/ba57de491619c466d99c5490d3df232b08b35823" alt="execution of a WHERE statement"
↑ Execution of a WHERE statement
In ”Translating Basic Excel to SQL” I also discussed the use of aggregate functions and GROUP BY statements. Using publicly available Airbnb data in New York City (original data source can be found here) say, for example, we wanted to see the mean price per listing by neighborhood group. We would use the code below to accomplish this task.
Here is the execution of the code above in DbVisualizer.
data:image/s3,"s3://crabby-images/3cb2d/3cb2dfc2c993a28f969ed2407ea0dd0c7de6480b" alt="execution of the code"
Filtering Values From Aggregate Functions
Now, say that we want to identify neighbourhood groups and associated mean listing prices for those neighbourhood groups where the mean listing price is less than $100.00. A first, and reasonable, guess on obtaining the desired output would be to use WHERE to filter.
Let’s see what happens when we execute this code.
data:image/s3,"s3://crabby-images/35bf0/35bf02d9b84ac0023a749fb05367a282eb510895" alt=""
We get an error message and the code in the editor is underlined with wavy red lines, which are present to help draw attention to erroneous code. So why is this code erroneous? Well, WHERE applies to non-aggregated records, but we want to filter mean price values, which are aggregated across neighbourhood groups. Now you are probably wondering if there is a way to filter aggregated values and the answer is yes. Even better, the answer is simple! Just replace WHERE with HAVING.
data:image/s3,"s3://crabby-images/ce3ef/ce3ef3c1c998cc5390cb875000826a757be8c96b" alt=""
Now the code is properly executed.
Conclusion
Being able to filter data is a fundamental skill for data analyses. When filtering in SQL, it is important to be mindful of where values in the filtering logic originated. That is, do the values on which we want to apply a filter represent the most granular level of detail in the particular data table of interest, or do the values represent a higher level of aggregation produced by an aggregate function and GROUP BY statement? Knowing this distinction and knowing the proper tools to use in both cases, that is, WHERE vs HAVING, can save a lot of headaches when performing data analyses in SQL.
data:image/s3,"s3://crabby-images/d31b7/d31b7e4b6a8605270848ac6f5ab1524bc53c729c" alt="Scott A. Adams."