FILTER Clause on Aggregates in SQL
The FILTER clause attaches a row-level condition directly to an aggregate function, restricting which rows that specific aggregate counts or sums, while leaving other aggregates in the same SELECT list unaffected.
Before this Conditional Aggregation (CASE inside Aggregates)
The FILTER clause attaches a condition directly to an aggregate function, restricting which rows that specific aggregate counts or sums, while leaving every other aggregate in the query unaffected.
The scenario: you're building a regional order dashboard. For each region, you need total orders, completed orders, and completed revenue — all in one row. You already know how to do this with CASE WHEN inside an aggregate. FILTER is a cleaner way to express the same thing. Instead of embedding a CASE expression inside the function, you place the condition after it:
SELECT
COUNT(*) AS total_orders,
COUNT(*) FILTER (WHERE status = 'delivered') AS delivered_orders,
SUM(total_amount) FILTER (WHERE status = 'delivered') AS delivered_revenue
FROM ordersEach FILTER is independent. COUNT(*) FILTER (WHERE status = 'delivered') only counts delivered orders. The plain COUNT(*) still counts everything. A row excluded by one FILTER is still visible to all other aggregates in the same SELECT list.
The CASE WHEN equivalent for the completed count:
COUNT(CASE WHEN status = 'completed' THEN 1 END)Both produce the same number. FILTER is just easier to read — the condition sits right next to the function it constrains, instead of being buried inside a CASE expression wrapped around the argument.
FILTER works with every standard aggregate: COUNT, SUM, AVG, MIN, MAX, and others. When you have several conditional aggregates in one SELECT list, each targeting a different condition, FILTER makes the query much easier to scan:
SELECT
COUNT(*) FILTER (WHERE status = 'completed') AS completed,
COUNT(*) FILTER (WHERE status = 'pending') AS pending,
COUNT(*) FILTER (WHERE status = 'cancelled') AS cancelled
FROM ordersThe same query with CASE WHEN would have three nested expressions to parse. With FILTER, each line states its condition plainly.
The one thing that trips people up
FILTER excludes rows from the aggregate entirely. CASE WHEN excludes them by substituting NULL, which COUNT then ignores. For COUNT, the result is the same. For AVG, it can differ in subtle ways — FILTER removes the row from both the sum and the count, while CASE WHEN removes the value from the sum but not necessarily the count, depending on how it's written. For standard AVG(CASE WHEN ... END), behavior matches FILTER. Just be deliberate about which rows you want included in each calculation.
When to use FILTER vs CASE WHEN
For a SELECT list with several conditional aggregates targeting different conditions, FILTER is easier to scan — each function states its condition plainly on one line. For a single conditional aggregate embedded inside a complex expression, CASE WHEN often integrates more naturally. Both are correct; the choice is about which makes the query easier for someone else to read. On teams that use PostgreSQL, FILTER is increasingly the standard for multi-condition breakdowns.
9 FILTER Clause on Aggregates practice problems
Write a query to return the total number of orders, the number of delivered orders, and the number of pending orders as a single row.
Write a query to return every customer ID, the total number of orders they have placed, and the number of those orders with `status = 'delivered'`.
Write a query to return every department ID, the total number of employees assigned to it, and the count of employees currently active.
Write a query to return every customer ID, their total order count, the number of delivered orders, and the total revenue from delivered orders only.
Write a query to return every category ID, the total number of products in that category, the count of products with `price` greater than `$100`, and the average `price` among products with `price` greater than `$100`.
Write a query to return every user ID, the user's total event count, the count of `event_type = 'purchase'` events, and the count of `event_type = 'checkout'` events.
Write a query to return every customer ID, their total order count, and the number of orders that are both delivered and have `total_amount` greater than `$200`.
Write a query to return every country, the total number of customers in that country, the count of customers with a missing `city`, and the count of customers with a recorded `city`.
Write a query to return every category ID, the total product count, the count of products priced above `$1,000`, and the combined `price` of those luxury products.
These problems are part of the FILTER Clause on Aggregates lesson in SQLMaxx, with instant grading and a worked solution on each.
Reading explains SQL. Writing it, over and over with instant feedback, is what makes you fluent.
That's the whole SQLMaxx loop: 600+ real problems, instant AI feedback, mastery you can actually see, and spaced review that won't let you forget.
Real problems. Not toy examples.
615 hand-built problems spanning all 66 concepts, from basic SELECTs to window functions, built on real schemas and real business questions, the kind you'll actually get asked on the job. Enough reps to make SQL automatic.
Write a query. Know if it's right in one second.
No copying an answer and hoping it clicked. The AI grader checks your real query against real data, catches exactly what's wrong, and explains the fix in plain English, like a senior analyst reading over your shoulder on every problem.
Stop guessing whether you actually know it.
SQLMaxx tracks every concept and shows you what you've mastered and what's still shaky. Your skills fill in one concept at a time, so 'I think I get joins' becomes something you can prove.
Learn it once. Keep it for good.
Most of what you learn this week fades by next week. So when a concept comes due for review, SQLMaxx hands you a fresh problem to solve from a blank editor, not a flashcard to re-read. A research-backed spaced-repetition algorithm (FSRS) times each return for right before you'd forget, so your SQL is still there months later, when the interview or the job actually needs it.
Practice, feedback, mastery, review. That's the loop that turns reading into real skill.
Start freeNo account, no credit card. Start solving in under a minute.