题目的意思是,在Employee表中找到第二大的Salary字段,然后以别名“SecondHighestSalary ”的形式输出
解题思路:
本题有两种思路。
一种是找到找到最大值,然后我们取比最大值小一点的最大值。
SQL:
SELECT Max(Salary) as SecondHighestSalary FROM Employee WHERE Salary< ( SELECT Max(Salary) FROM Employee);
Max()函数表示返回最大值,如果没有则返回NULL。
Salary<( SELECT Max(Salary) FROM Employee) 则是子查询的概念。
例如 Select * From Table_A Where key in (a,b); 如果其中(a,b)的数据来源于Table_B或者Table_A表。我们就可以使用子查询,将(a,b)的数据替换为我们查询到的数据。
Select * From Table_A Where key in (Select a,b From Table_B Where Inr=‘XXXXXXX‘);
“as XXX”则表示将输出的数据以“XXX”为别名的方式输出。
还有一种方法就是,我们将数据进行排序,然后取第二位数。
这种方法的难点在于:1、数据可能会有重复项。
2、需要先排序再取数,所以不能用between 方法因为between方法必须要在order by 之前。
3、整个表都是同一个数据的极端情况。
要解决以上3个难点。我们首先使用DISTINCT方法将数据的重复项过滤只剩一条,然后使用LIMIT分页的方法取第二个值,对于情况3中的极端情况,我们使用 IFNULL来进行判断,如果没取到值就返回NULL。需要注意的是,LIMIT是MySQL中的用法,Oracle是不支持LIMIT的。
SQL:
SELECT IFNULL( (SELECT distinct Salary FROM Employee order by Salary desc limit 1,1),null)as SecondHighestSalary;
其中LIMIT 的用法是这样的:
LIMIT m,n "m"表示偏移量,表示从第几位开始取。"n"表示长度,即需要取多少位。需要注意的是偏移量的第一位为0,所以"LIMIT 1,1"表示从第二位开始取长度为1的数也就是第二位数。"LIMIT 0,n"或者"LIMIT n "则表示从第一位开始取,一直取到第n位为止。"LIMIT m,-1"表示从第m位开始取,一直取到表的末尾。
distinct 表示将查询到的结果过滤重复项,只保留一项。
IFNULL(Exp1,Exp2),类似一个三元表达式,如果Exp1不为空则返回Exp1的结果集,如果为空就返回Exp2.
方法二的关键点在于排序和取数,排序我们除了使用ORDER BY 方法之外,还可以使用比较的方式排序。
SQL:
SELECT DISTINCT(e2.Salary) FROM Employee e1,Employee e2 WHERE e2.Salary>=e1.Salary;
使用“>=”的方式,是为了避免Salary字段有0值而导致无法比较。
(sql空值表示:
在允许空值的情况下,空值是NULL,指没有填写过数据;在不允许为空的情况下,数值型字段默认为0)
需要注意的是,这种排序方式非常非常的消耗系统资源,在实际生产中使用这种方式来排序是不负责任的行为(见https://www.cnblogs.com/MarsDing/p/9856272.html),所以如非必要尽量不要使用这种方式而使用更快捷消耗更小的的Order by。
现在已经排好序了,那么接下来我们要进行取数。取数可以使用上面的LIMIT方法。
LIMIT方法:
SELECT IFNULL((SELECT DISTINCT(e2.Salary) FROM Employee e1,Employee e2 WHERE e2.Salary>e1.Salary LIMIT 1,1),NULL) as SecondHighestSalary;
即,将:order by Salary desc 替换为 WHERE e2.Salary>e1.Salary
除此之外还有其他方法吗?我们知道,SELECT语句实际上是个循环语句,既然是个循环语句而且已经排好序,那么也就是说如果我们要取第二大的数据,我们可以取当他循环到第二次时的那一行。怎么取到那一行呢。学过For循环都知道,要先对这个循环进行计数,然后循环到我们需要的那一次再取值。那么SQL中怎么计数呢。答案是使用COUNT方法。
COUNT方法
SELECT e1.Salary FROM Employee e1 WHERE 2=(Select Count(distinct(e2.Salary)) from Employee e2 where e2.Salary >= e1.Salary) as SecondHighestSalary;
上面两种方法也可以用于,取任意第N位数据时。
COUNT方法之所以不能使用Order by,是因为Order by是对结果进行排序,而我们需要的是已排序好的结果。
如果要使用Order by 方法,需要再加一层子查询。
COUNT-Order By 方法
SELECT Salary FROM Employee WHERE 2=(Select Count(distinct(Salary)) from (Select Salary From Employee order by Salary desc)) as SecondHighestSalary;
.
由于Oracle中不能使用LIMIT,所以我们用rownum来处理此问题
Select Salary from (select amt ,rownum n From (Select distinct amt from Employee order by Salary desc)) where n=2;
其中的n=2可以替换为任意正整数,即为第N位。
总结:对一个题要通过不同角度思考并优化解答
转载于:https://www.cnblogs.com/contixue/p/7057025.html
【LeetCode刷题】SQL-Second Highest Salary 及扩展以及Oracle中的用法
标签:难点 取数 des 分页 xxxxx 必须 strong 循环语句 row
小编还为您整理了以下内容,可能对您也有帮助:
查询数据库中哪些表空间的空闲空间百分比低于20%
本人对oracle了解有限,
以下摘自:http://blog.csdn.net/ecast/article/details/6911535,希望能够帮到你。
--1查看表空间已经使用的百分比
Sql代码 select a.tablespace_name,a.bytes/1024/1024 "Sum MB",(a.bytes-b.bytes)/1024/1024 "used MB",b.bytes/1024/1024 "free MB",round(((a.bytes-b.bytes)/a.bytes)*100,2) "percent_used"
from
(select tablespace_name,sum(bytes) bytes from dba_data_files group by tablespace_name) a,
(select tablespace_name,sum(bytes) bytes,max(bytes) largest from dba_free_space group by tablespace_name) b
where a.tablespace_name=b.tablespace_name
order by ((a.bytes-b.bytes)/a.bytes) desc
“Sum MB”表示表空间所有的数据文件总共在操作系统占用磁盘空间的大小
比如:test表空间有2个数据文件,datafile1为300MB,datafile2为400MB,那么test表空间的“Sum MB”就是700MB
“userd MB”表示表空间已经使用了多少
“free MB”表示表空间剩余多少
“percent_user”表示已经使用的百分比
--2比如从1中查看到MLOG_NORM_SPACE表空间已使用百分比达到90%以上,可以查看该表空间总共有几个数
据文件,每个数据文件是否自动扩展,可以自动扩展的最大值。
Sql代码 select file_name,tablespace_name,bytes/1024/1024 "bytes MB",maxbytes/1024/1024 "maxbytes MB" from dba_data_files
where tablespace_name='MLOG_NORM_SPACE';
--2.1 查看 xxx 表空间是否为自动扩展
Sql代码 select file_id,file_name,tablespace_name,autoextensible,increment_by from dba_data_files order by file_id desc;
--3比如MLOG_NORM_SPACE表空间目前的大小为19GB,但最大每个数据文件只能为20GB,数据文件快要写满,可以增加表空间的数据文件
用操作系统UNIX、Linux中的df -g命令(查看下可以使用的磁盘空间大小)
获取创建表空间的语句:
Sql代码 select dbms_metadata.get_ddl('TABLESPACE','MLOG_NORM_SPACE') from al;
--4确认磁盘空间足够,增加一个数据文件
Sql代码 alter tablespace MLOG_NORM_SPACE
add datafile '/oracle/oms/oradata/mlog/Mlog_Norm_data001.dbf'
size 10M autoextend on maxsize 20G
--5验证已经增加的数据文件
Sql代码 select file_name,file_id,tablespace_name from dba_data_files
where tablespace_name='MLOG_NORM_SPACE'
--6如果删除表空间数据文件,如下:
Sql代码 alter tablespace MLOG_NORM_SPACE
drop datafile '/oracle/oms/oradata/mlog/Mlog_Norm_data001.dbf'