oracle sql查询缺失号,Oracle层次查询和分析函数

摘要

一组连续的数,去掉中间一些数,如何求出剩下的数的区间(即号段)?知道号段的起止,如何求出该号段内所有的数?知道一个大的号段范围和已经取过的号段,如何求出可用的号段?利用Oracle提供的强大的查询功能以及分析函数,我们可以很轻松的解决上述问题。

n关键词:

号段选取、连续数、断点、层次查询、分析函数、connect by、rownum、level、lead、lag

1.问题的提出

在实际工作中,我们常常会碰到号段选取的问题,例如:

n一组连续的数,去掉中间一些数,要求出剩下的数的区间(即号段)例如:一串数字为1,2,3,4,7,9,10,则号段为1-4,7-7,9-10

n知道号段的起止,要求出该号段内所有的数例如:号段为1-3,15-15,则号段内所有的数为1,2,3,15

n一组数,中间可能有断点,要求出缺失的数例如:一串数字为1,2,3,4,7,9,10,则缺失的数为5,6,8

n已知大号段范围及已用号段范围,求可用号段范围例如:大号段范围0-999,已用号段范围0-200,399-599,则可用号段范围为201-398,600-999

2.基础知识

先做下热身运动,回顾一下层次查询和lead/lag函数的运用。

2.1伪列rownum和level

伪列就是并非在表中真正存在的列。已有很多资料介绍rownum和level这两个伪列。这里只想强调一点,伪列是只针对结果集的。

2.2利用层次查询构造连续的数

n产生5~8这4个连续的数

select * from (select rownum+4 from dual connect by rownum<5);

select * from (select level+4 from dual connect by level<5);

n以8月为界,例如2005年8月1日,之前的在校学生入学年份为2001~2004,之后的为2002~2005。求当前日期下的在校学生入学年份:

select * from (select to_char(add_months(sysdate, 4), 'yyyy') - rownum from dual connect by rownum<5);

2.3用分析函数Lead和Lag获得相邻行的字段值

select rn, lag(rn)over(order by rn) previos, lead(rn)over(order by rn) next from (select rownum+4 rn from dual connect by rownum<5);

RNPREVIOSNEXT

---------- ---------- ----------

56

657

768

87

简单的说,在这里,Lag是获得前一行的内容,而Lead是获得后一行的内容。

select rn, lag(rn,2,-1)over(order by rn) previos, lead(rn,2,-1) over(order by rn) next from (select rownum+4 rn from dual connect by rownum<5);

RNPREVIOSNEXT

---------- ---------- ----------

5-17

6-18

75-1

86-1

这里,通过指定offset参数来获得两行前的内容和两行后的内容,如果offset超出范围并且未设定默认值-1,那么系统会自动将其值设为NULL。

3.问题的解决

有了基础知识的积累,我们就可以解决前面提到的问题。

3.1已知号码求号段

3.1.1题例

我有一个表结构,

fphm,kshm

2014,00000001

2014,00000002

2014,00000003

2014,00000004

2014,00000005

2014,00000007

2014,00000008

2014,00000009

2013,00000120

2013,00000121

2013,00000122

2013,00000124

2013,00000125

(第二个字段内可能是连续的数据,可能存在断点。)

怎样能查询出来这样的结果,查询出连续的记录来。

就像下面的这样?

2014,00000001,00000005

2014,00000009,00000007

2013,00000120,00000122

2013,00000124,00000125

3.1.2解答

思路:利用lag取得前一行的kshm,然后和本行的kshm想比,如果差值为1,说明这一行和上一行是连续的。由于首尾的特殊性,故而需要先用max和min来获得首尾点。

select fphm, nvl(lag(e)over(partition by fphm order by s),minn) ST, nvl(S,maxn) EN from

(select fphm, lag(kshm,1) over(partition by fphm order by kshm) S, kshm E, min(kshm)over(partition by fphm) minn, max(kshm) over(partition by fphm) maxn from t)

where nvl(E-S-1,1)<>0;

FPHM STEN

---------- ---------- ----------

20130000012000000122

20130000012400000125

20140000000100000005

20140000000700000009

3.2根据号段求出包含的数

3.2.1题例

有表及测试数据如下:

CREATE TABLE T20

(

ID NUMBER(2),

S NUMBER(5),

E NUMBER(5)

);

INSERT INTO T20 ( ID, S, E ) VALUES ( 1, 10, 11);

INSERT INTO T20 ( ID, S, E ) VALUES ( 2, 1, 5);

INSERT INTO T20 ( ID, S, E ) VALUES ( 3, 88, 92);

COMMIT;

S为号段起点,E为号段终点,求出起点和终点之间的数(包括起点和终点)

3.2.2解答

很明显,这需要构造序列来解决问题

select a.id, a.s, a.e,b.dis, a.S+b.dis-1 h from

t20 a,

(select rownum dis from

(select max(e-s)+1 gap from t20)

connect by rownum<=gap) b

where a.e>=a.s+b.dis-1

order by a.id, 4

运行结果:

IDSEDISH

---------- ---------- ---------- ---------- ----------

11011110

11011211

21511

21522

21533

21544

21555

38892188

38892289

38892390

38892491

38892592

我们再看下面这种做法:

select a.id, a.s, a.e,rownum, a.S+rownum-1 h from

t20 a ,

(select id, e-s+1 gap from t20 where id=2) b

where a.id=b.id

connect by rownum<=gap

IDSE ROWNUMH

---------- ---------- ---------- ---------- ----------

21511

21522

21533

21544

21555

嗯,得到的结果也是正确的,若我们把粗斜体字部分去掉后,看看结果是什么样:

IDSE ROWNUMH

---------- ---------- ---------- ---------- ----------

11011110

11011211

21533

21544

21555

21566

388 92794

这样的结果,显然不是我们需要的,更何况,这是错误的。由此更能深入理解,伪列是只针对结果集的。

3.3求缺失的号

3.3.1题例

table T,列:serial_no

我想能够查询一下serial_no这个字段的不连续的值。

例如:

serial_no

1

2

3

4

6

8

9

10

我想一个sql语句查出来缺失的号码,

显示结果为:

5

7

3.3.2解答

思路:找出数B和它前面的数A进行比较(数按从大到小进行排序),如果B-A=1,则说明是连续的,中间没有断点。

select distinct s+level-1 rlt from (select lag(serial_no,1) over(order by serial_no)+1 S, serial_no-1 E from t) where E-S<>0 connect by level<=e-s

3.4求尚未使用的号段

3.4.1题例

表A结构:

bill_type_id varchar2(1),

bill_start number,

bill_end number,

office_level varchar2(4)

数据如下:

A 0 999 1

A 0 199 2

A 300 499 2

A 700 799 2

sql目的是取出包含在level1级别里的,还没有录入level2级别的号段。

3.4.2解答

这个好像是3.1和3.3这两个问题的逆问题

创建表及测试数据:

CREATE TABLE T8

(

A NUMBER(4),

B NUMBER(4),

C NUMBER(4),

Q VARCHAR2(1 BYTE)

);

Insert into T8(A, B, C, Q)Values(555, 666, 2, 'A');

Insert into T8(A, B, C, Q)Values(100, 199, 2, 'A');

Insert into T8(A, B, C, Q)Values(0, 999, 1, 'A');

Insert into T8(A, B, C, Q)Values(300, 499, 2, 'A');

COMMIT;

思路:将大号段的边界与小号段的边界相比,从大号段中将小号段“挖”掉,这样剩下的就是可用号段了。

select S,E from

(

SELECT NVL2(LAG(A)OVER(PARTITION BY Q ORDER BY A), B+1, MIN(A)OVER(PARTITION BY Q)) S,

NVL(LEAD(A)OVER(PARTITION BY Q ORDER BY A)-1, MAX(B)OVER(PARTITION BY Q)) E

from t8 START WITH C=1 CONNECT BY C-1 = PRIOR C AND Q= PRIOR Q

)

where s<=e

运行结果:

SE

---------- ----------

099

200299

500554

667999

参考资料

n参考资料

uOracle 8i SQL Reference

uhttp://lastwinner.itpub.net

uhttp://hmxxyy.itpub.net/

n论坛相关讨论帖子

u请教查询语句的写法?http://www.itpub.net/435578.html

u如何sql查询出连续号码段http://www.itpub.net/354052.html

u请教关于号段选取的sql写法http://www.itpub.net/480536.html

u知道号段起止,如何选择该号段内的所有号码?http://www.itpub.net/701508.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值