I have a query to compute the 3rd highest salary for an employee as shown below.
SELECT MAX(salary)
FROM employee
WHERE emp_no = 1
AND salary < (SELECT MAX(salary)
FROM employee
WHERE emp_no = 1
AND salary NOT IN (SELECT MAX(salary)
FROM employee
WHERE emp_no = 1))
How can I apply this query to give the 3rd highest salary for each employee which can be fetched by the query
select distinct(emp_no)
from employee
Note: without using special functions like dense_rank()
sample table
--------------------
EMP SALARY
--------------------
1 1000
1 1000
1 900
1 800--->Required
2 1000
2 1000
2 500
2 400---->Required
4
Answers
The alternative way without using dense_rank:
You are looking for each employee’s third highest salary. It can happen that we find the same salary for an employee multiple times in the table, as your sample data shows. So, make the salaries per employee distinct, then rank the rows with
ROW_NUMBER
,RANK
orDENSE_RANK
(which doesn’t matter with distinct values) and then pick the third ranked.An alternative would be to count higher salaries in a subquery and select those salaries where exist two higher salaries for the employee:
Demo: https://dbfiddle.uk/?rdbms=postgres_14&fiddle=1e17669870f2e9c7f5867bf2ee6c24bf
I think you cannot do this without analytic function, because database have to analyze data between rows and order it in memory for given sub-ordered results
This seems possible with a lateral join:
Online example
But a window function would be much more efficient.