LeetCode 577 - Employee Bonus¶
Database Language: SQL Server
Difficulty:
Problem Description¶
Input¶
Table: Employee¶
Column Name | Type |
---|---|
empId | int |
name | varchar |
supervisor | int |
salary | int |
empId is the column with unique VALUES for this table. Each row of this table indicates the name and the ID of an employee in addition to their salary and the id of their manager.
Table: Bonus¶
Column Name | Type |
---|---|
empId | int |
bonus | int |
empId is the column of unique VALUES for this table. empId is a foreign key (reference column) to empId from the Employee table. Each row of this table contains the id of an employee and their respective bonus.
Requirement¶
Write a solution to report the name and bonus amount of each employee with a bonus less than 1000.
Return the result table in any order.
The result format is in the following example.
Examples¶
Example 1¶
Input¶
Employee table:
empId | name | supervisor | salary |
---|---|---|---|
3 | Brad | null | 4000 |
1 | John | 3 | 1000 |
2 | Dan | 3 | 2000 |
4 | Thomas | 3 | 4000 |
Bonus table:
empId | bonus |
---|---|
2 | 500 |
4 | 2000 |
Output¶
name | bonus |
---|---|
Brad | null |
John | null |
Dan | 500 |
SQL Schema¶
CREATE TABLE [dbo].[Employee] (empId INT PRIMARY KEY, name VARCHAR(255), supervisor INT, salary INT);
CREATE TABLE [dbo].[Bonus] (empId INT PRIMARY KEY, bonus INT);
TRUNCATE TABLE [dbo].[Employee];
INSERT INTO [dbo].[Employee] (empId, name, supervisor, salary) VALUES ('3', 'Brad', NULL, '4000');
INSERT INTO [dbo].[Employee] (empId, name, supervisor, salary) VALUES ('1', 'John', '3', '1000');
INSERT INTO [dbo].[Employee] (empId, name, supervisor, salary) VALUES ('2', 'Dan', '3', '2000');
INSERT INTO [dbo].[Employee] (empId, name, supervisor, salary) VALUES ('4', 'Thomas', '3', '4000');
TRUNCATE TABLE [dbo].[Bonus];
INSERT INTO [dbo].[Bonus] (empId, bonus) VALUES ('2', '500');
INSERT INTO [dbo].[Bonus] (empId, bonus) VALUES ('4', '2000');
Solution¶
Given that there are 2 tables provided and data from each table is needed to come out of the output, this means that a JOIN will be used to join the Employee
table with the Bonus
table. Looking at the Bonus
table, it can be seen that not all employees have an entry in this table. Based on the sample output, if an employee does not have an entry in the Bonus
table, then that employee should be included in the output. From this information, we can determine that a LEFT OUTER JOIN
will be used instead of an INNER JOIN
. If the output will not include those employees who do not have a record in the Bonus
table, then the INNER JOIN
will be used. But since the output also wants those employees who do not have bonus aside from those who had less than 1000, then the LEFT OUTER JOIN
will be used with the Employee
table on the left side of the join. So the join clause will look as follows:
FROM Employee LEFT OUTER JOIN Bonus
ON Employee.empId = Bonus.empId
Since we are only interested on those employees with no bonus at all (Bonus.bonus IS NULL
) and those who got less than 1000 bonus (Bonus.bonus < 1000
), then we can add those conditions in the WHERE
clause:
FROM Employee LEFT OUTER JOIN Bonus
ON Employee.empId = Bonus.empId
WHERE Bonus.bonus IS NULL OR Bonus.bonus < 1000
Lastly, the requirement is to return just the name of the employee as well as the bonus received (if any):
# Final Solution Query
SELECT Employee.name, Bonus.bonus
FROM Employee LEFT OUTER JOIN Bonus
ON Employee.empId = Bonus.empId
WHERE Bonus.bonus IS NULL OR Bonus.bonus < 1000
The query plan generated by SQL Server for the query above is as follows:
|--Filter(WHERE:([leetcode].[dbo].[Bonus].[bonus] IS NULL OR [leetcode].[dbo].[Bonus].[bonus]<(1000)))
|--Nested Loops(Left Outer Join, WHERE:([leetcode].[dbo].[Employee].[empId]=[leetcode].[dbo].[Bonus].[empId]))
|--Clustered Index Scan(OBJECT:([leetcode].[dbo].[Employee].[PK_Employee]))
|--Clustered Index Scan(OBJECT:([leetcode].[dbo].[Bonus].[PK_Bonus]))
And here's the faster runtime for this query:
- Runtime: 591ms
- Beats: 98.12% as of July 16, 2024