好几天没有认真刷题了,这两天猛刷了一把SQL题目。
而后用hexo搭建了本身的BLOG,还在摸索中,后续渐渐的就会两边都同步文章。git
leetcode上对于数据库是有单独的19题的,我如今的进度是8/19,刷的仍是有点慢,并且不少地方效率不高,还得作n刷处理。
毕竟后续若是考虑到要说数据分析的话,取数上的效率也得保证。github
175. 组合两个表
难度:简单sql
表1: Person数据库
列名 | 类型 |
---|---|
PersonId | int |
FirstName | varchar |
LastName | varchar |
PersonId 是上表主键
表2: Addresshexo
列名 | 类型 |
---|---|
AddressId | int |
PersonId | int |
City | varchar |
State | varchar |
AddressId 是上表主键网站
编写一个 SQL 查询,知足条件:不管 person 是否有地址信息,都须要基于上述两表提供 person 的如下信息:
FirstName, LastName, City, State
个人题解:code
SELECT Person.FirstName,Person.LastName,Address.City,Address.State From Person Left Join Address ON Person.PersonId = Address.PersonId
解题思路:
由于不管address可能为空,因此用left join的方式,加入Address表。排序
其余:
好久没有用过left join,有些概念有点忘记,顺便来复习下知识点。
在left join
以前的左表是会被彻底返回的,哪怕left join的右表没有对应的数据。leetcode
select * from table_1 left join table_2
这里的话会返回全部table_1
的行。get
sql的left join 、right join 、inner join之间的区别:
- left join(左联接)
返回包括左表中的全部记录和右表中联结字段相等的记录
- right join(右联接)
返回包括右表中的全部记录和左表中联结字段相等的记录
- inner join(等值链接)
只返回两个表中联结字段相等的行
176. 第二高的薪水
难度:简单
编写一个 SQL 查询,获取 Employee 表中第二高的薪水(Salary) 。 | |
---|---|
Id | Salary |
1 | 100 |
2 | 200 |
3 | 300 |
例如上述 Employee 表,SQL查询应该返回 200 做为第二高的薪水。若是不存在第二高的薪水,那么查询应返回 null。
SecondHighestSalary |
---|
200 |
个人题解:
SELECT MAX(Salary) AS SecondHighestSalary FROM Employee WHERE Salary < (SELECT MAX(Salary) FROM Employee)
解题思路:
使用max()
来获取两次最大值,由于是同一张表,小于最大值的“最大值”就是第二大的值了。
其余:
通常主要查找最大值,这题查找的是第二大的值。
主要是思路上要调整下,通常程序语言上会作排序。
SQL里面也能够考虑用排序试下,若是要取第二条数据的话,就得先取前两条数据,再倒序取第一条。
181. 超过经理收入的员工
难度:简单
Employee 表包含全部员工,他们的经理也属于员工。每一个员工都有一个 Id,此外还有一列对应员工的经理的 Id。 | |||
---|---|---|---|
Id | Name | Salary | ManagerId |
1 | Joe | 70000 | 3 |
2 | Henry | 80000 | 4 |
3 | Sam | 60000 | NULL |
4 | Max | 90000 | NULL |
给定 Employee 表,编写一个 SQL 查询,该查询能够获取收入超过他们经理的员工的姓名。在上面的表格中,Joe 是惟一一个收入超过他的经理的员工。
Employee |
---|
Joe |
个人题解:
SELECT p1.Name AS Employee FROM Employee p1,Employee p2 WHERE p1.ManagerId = p2.Id AND p1.Salary > p2.Salary
解题思路:
查询两次同一张表,主条件为匹配经理Id和用户Id,再作比对大小。
其余:
对于同一张表查询两次,其实应该验证下效率到底如何,检查下是否有更快的查询方案。
182. 查找重复的电子邮箱
难度:简单
编写一个 SQL 查询,查找 Person 表中全部重复的电子邮箱。
示例: | |
---|---|
Id | |
1 | a@b.com |
2 | c@d.com |
3 | a@b.com |
根据以上输入,你的查询应返回如下结果:
a@b.com |
说明:全部电子邮箱都是小写字母。
个人题解:
SELECT distinct(p1.Email) from Person p1,Person p2 where p1.Email = p2.Email AND p1.Id != p2.Id
解题思路:
仍是查询同一张表两次,而后使用distinct,只输出单个结果。
其余:distinct
用于返回惟一不一样的值。
有distinct的字段必须放在开头。
183. 从不订购的客户
难度:简单
某网站包含两个表,Customers 表和 Orders 表。编写一个 SQL 查询,找出全部从不订购任何东西的客户。
Customers 表:
Id | Name |
---|---|
1 | Joe |
2 | Henry |
3 | Sam |
4 | Max |
Orders 表: | |
---|---|
Id | CustomerId |
1 | 3 |
2 | 1 |
例如给定上述表格,你的查询应返回:
Customers |
---|
Henry |
Max |
个人题解:
SELECT c.name AS Customers FROM Customers c WHERE c.Id Not in(SELECT CustomerId FROM Orders)
解题思路:
取出Order表的数据,而后和Customers的Id作校验。
其余:
若是不是用取出Customers的ID来作比较的,就是Id!=CusomerId,而是查询两张表直接输出结果的话,会把每次的不对应的结果都输出。由于等于两张表都被完整比对过一次。
184. 部门工资最高的员工
难度:中等
Employee 表包含全部员工信息,每一个员工有其对应的 Id, salary 和 department Id。 | |||
---|---|---|---|
Id | Name | Salary | DepartmentId |
1 | Joe | 70000 | 1 |
2 | Henry | 80000 | 2 |
3 | Sam | 60000 | 2 |
4 | Max | 90000 | 1 |
Department 表包含公司全部部门的信息。 | |
---|---|
Id | Name |
1 | IT |
2 | Sales |
编写一个 SQL 查询,找出每一个部门工资最高的员工。例如,根据上述给定的表格,Max 在 IT 部门有最高工资,Henry 在 Sales 部门有最高工资。
Department | Employee | Salary |
---|---|---|
IT | Max | 90000 |
Sales | Henry | 80000 |
个人题解:
select d.Name as Department,e.Name as Employee,e.Salary from Department d,Employee e where e.DepartmentId = d.ID and e.Salary = (select max(Salary) from Employee where d.id = DepartmentId)
解题思路:
这题参考了其余人的思路,后续须要本身再写一次。
实际上是转了两次弯,第一次是根据部门Id查询出每一个部门最高的薪水,再根据这个薪水找到对应的人。
其余:
196. 删除重复的电子邮箱
难度:简单
编写一个 SQL 查询,来删除 Person 表中全部重复的电子邮箱,重复的邮箱里只保留 Id 最小 的那个。 | |
---|---|
Id | |
1 | john@example.com |
2 | bob@example.com |
3 | john@example.com |
Id 是这个表的主键。
例如,在运行你的查询语句以后,上面的 Person 表应返回如下几行: | |
---|---|
Id | |
1 | john@example.com |
2 | bob@example.com |
个人题解:
DELETE p1 FROM Person p1,Person p2 WHERE p1.Email = p2.Email and p1.Id > p2.Id
解题思路:
这题一开始也有点被绕住了,后面渐渐作多了两次查询同步一张表就还好,
核心思路就是查询相同的值,且Id不一样,咱们delete的是Id较大的那一行。
其余:
Null.
596. 超过5名学生的课
难度:简单
有一个courses 表 ,有: student (学生) 和 class (课程)。
请列出全部超过或等于5名学生的课。
例如,表: | |
---|---|
student | class |
A | Math |
B | English |
C | Math |
D | Biology |
E | Math |
F | Computer |
G | Math |
H | Math |
I | Math |
应该输出: |
---|
class |
Math |
个人题解:
select class From course group by class having count(class)>=5
解题思路:
对课程进行分组,分组后记数大于等于5的就取出数值。
其余
Null
学生在每一个课中不该被重复计算。
个人题解:
SELECT class FROM (select distinct * from courses) as new GROUP BY class HAVING count(*) >= 5
解题思路:
其余: