DB2公共表表达式WITH语句的使用全解.docx

上传人:b****1 文档编号:14024 上传时间:2022-09-30 格式:DOCX 页数:11 大小:16.05KB
下载 相关 举报
DB2公共表表达式WITH语句的使用全解.docx_第1页
第1页 / 共11页
DB2公共表表达式WITH语句的使用全解.docx_第2页
第2页 / 共11页
DB2公共表表达式WITH语句的使用全解.docx_第3页
第3页 / 共11页
DB2公共表表达式WITH语句的使用全解.docx_第4页
第4页 / 共11页
DB2公共表表达式WITH语句的使用全解.docx_第5页
第5页 / 共11页
点击查看更多>>
下载资源
资源描述

DB2公共表表达式WITH语句的使用全解.docx

《DB2公共表表达式WITH语句的使用全解.docx》由会员分享,可在线阅读,更多相关《DB2公共表表达式WITH语句的使用全解.docx(11页珍藏版)》请在冰豆网上搜索。

DB2公共表表达式WITH语句的使用全解.docx

DB2公共表表达式WITH语句的使用全解

 

WITH语句(DB2公共表表达式的使用)

----start

   说起WITH语句,除了那些第一次听说WITH语句的人,大部分人都觉得它是用来做递归查询的。

其实那只是它的一个用途而已,它的本名正如我们标题写的那样,叫做:

公共表表达式(CommonTableExpression),从字面理解,大家觉得它是用来干嘛的呢?

其实,它是用来定义临时集合的。

啊?

VALUES语句不是用来定义临时集合的吗?

怎么WITH语句也用来定义临时集合呢?

它们有什么区别呢?

 

VALUES语句是用明确的值来定义临时集合的,如下:

[c-sharp] viewplaincopyprint?

1.VALUES (1,2), (1,3),(2,1)  

 

WITH语句是用查询(也就是select语句)来定义临时集合的,从这个角度讲,有点像视图,不过不是视图,大家千万别误解。

如下:

[c-sharp]viewplaincopyprint?

1.CREATE TABLE USER (  

2.NAME VARCHAR(20) NOT NULL,---姓名  

3.SEX INTEGER,---性别(1、男   2、女)  

4.BIRTHDAY DATE---生日  

5.);  

[c-sharp] viewplaincopyprint?

1.WITH TEST(NAME_TEST, BDAY_TEST) AS  

2.(  

3.SELECT NAME,BIRTHDAY FROM USER--语句1  

4.)  

5.SELECT NAME_TEST FROM TEST WHERE BDAY_TEST='1949-10-1';--语句2  

 

下面我们来解释一下,首先语句1执行,它会产生一个有两列(NAME,BIRTHDAY)的结果集;接着,我们将这个结果集命名为test,并且将列名重命名为NAME_TEST,BDAY_TEST;最后我们执行语句2,从这个临时集合中找到生日是1949-10-1,也就是共和国的同龄人。

 

怎么样?

如果你感觉不好理解,请仔细的分析一下上面的语句。

下面我们举个VALUES语句和WITH语句结合使用的例子,如下:

[c-sharp]viewplaincopyprint?

1.WITH TEST(NAME_TEST, BDAY_TEST) AS  

2.(  

3.VALUES ('张三','1997-7-1'),('李四','1949-10-1')  

4.)  

5.SELECT NAME_TEST FROM TEST WHERE BDAY_TEST='1949-10-1'  

从上面的介绍和WITH语句不为大多数人所熟悉可以猜测,WITH语句是为复杂的查询为设计的,的确是这样的,下面我们举个复杂的例子,想提高技术的朋友可千万不能错过。

考虑下面的情况:

[c-sharp] viewplaincopyprint?

1.CREATE TABLE USER  

2.(  

3.NAME VARCHAR(20) NOT NULL,--姓名  

4.DEGREE INTEGER NOT NULL,--学历(1、专科 2、本科 3、硕士 4、博士)  

5.STARTWORKDATE date NOT NULL,--入职时间  

6.SALARY1 FLOAT NOT NULL,--基本工资  

7.SALARY2 FLOAT NOT NULL--奖金  

8.);  

 

假设现在让你查询一下那些1、学历是硕士或博士 2、学历相同,入职年份也相同,但是工资(基本工资+奖金)却比相同条件员工的平均工资低的员工。

(哈哈,可能是要涨工资),不知道你听明白问题没有?

该怎么查询呢?

我们是这样想的:

 

1、查询学历是硕士或博士的那些员工得到结果集1,如下:

[c-sharp] viewplaincopyprint?

1.SELECT NAME,DEGREE,YEAR(STARTWORKDATE) AS WORDDATE, SALARY1+SALARY2 AS SALARY FROM USER WHERE DEGREE IN (3,4);  

 

2、根据学历和入职年份分组,求平均工资得到结果集2,如下:

[c-sharp] viewplaincopyprint?

1.SELECT DEGREE,YEAR(STARTWORKDATE) AS WORDDATE, AVG(SALARY1+SALARY2) AS AVG_SALARY   

2.FROM USER WHERE DEGREE IN (3,4)  

3.GROUP BY DEGREE,YEAR(STARTWORKDATE)  

 

3、以学历和入职年份为条件 联合两个结果集,查找工资<平均工资 的员工,以下是完整的SQL:

[c-sharp] viewplaincopyprint?

1.WITH TEMP1(NAME,DEGREE,WORDDATE,SALARY) AS   

2.(  

3.SELECT NAME,DEGREE,YEAR(STARTWORKDATE) AS WORDDATE, SALARY1+SALARY2 AS SALARY FROM USER WHERE DEGREE IN (3,4)  

4.),  

5.TEMP2 (DEGREE,WORDDATE,AVG_SALARY) AS  

6.(  

7.SELECT DEGREE,YEAR(STARTWORKDATE) AS WORDDATE, AVG(SALARY1+SALARY2) AS AVG_SALARY   

8.FROM USER WHERE DEGREE IN (3,4)  

9.GROUP BY DEGREE,YEAR(STARTWORKDATE)  

10.)  

11.SELECT NAME FROM TEMP1, TEMP2 WHERE   

12.TEMP1.DEGREE=TEMP2.DEGREE   

13.AND TEMP1.WORDDATE=TEMP2.WORDDATE   

14.AND SALARY

 

查询结果完全正确,但我们还有改善的空间,在查询结果集2的时候,我们是从user表中取得数据的。

其实此时结果集1已经查询出来了,我们完全可以从结果集1中通过分组得到结果集2,而不用从uer表中得到结果集2,比较上面和下面的语句你就可以知道我说的是什么意思了!

[c-sharp] viewplaincopyprint?

1.WITH TEMP1(NAME,DEGREE,WORDDATE,SALARY) AS   

2.(  

3.SELECT NAME,DEGREE,YEAR(STARTWORKDATE) AS WORDDATE, SALARY1+SALARY2 AS SALARY FROM USER WHERE DEGREE IN (3,4)  

4.),  

5.TEMP2 (DEGREE,WORDDATE,AVG_SALARY) AS  

6.(  

7.SELECT DEGREE,WORDDATE, AVG(SALARY) AS AVG_SALARY   

8.FROM TEMP1  

9.GROUP BY DEGREE,WORDDATE  

10.)  

11.SELECT NAME FROM TEMP1, TEMP2 WHERE   

12.TEMP1.DEGREE=TEMP2.DEGREE   

13.AND TEMP1.WORDDATE=TEMP2.WORDDATE   

14.AND SALARY

 

可能有些朋友会说,我不用WITH语句也可以查出来,的确是这样,如下:

[c-sharp]viewplaincopyprint?

1.SELECT U.NAME FROM USER AS U,  

2.(  

3.SELECT DEGREE,YEAR(STARTWORKDATE) AS WORDDATE, AVG(SALARY1+SALARY2) AS AVG_SALARY   

4.FROM USER WHERE DEGREE IN (3,4)  

5.GROUP BY DEGREE,YEAR(STARTWORKDATE)  

6.) AS G  

7.WHERE U.DEGREE=G.DEGREE  

8.AND YEAR(U.STARTWORKDATE)=G.WORDDATE  

9.AND (SALARY1+SALARY2)

那使用WITH和不使用WITH,这两种写法有什么区别呢?

一般情况下这两种写法在性能上不会有太大差异,但是,

1、当USER表的记录很多

2、硕士或博士(DEGREEIN(3,4))在USER表中的比例很少

 

当满足以上条件时,这两种写法在性能的差异将会显现出来,为什么呢?

因为不使用WITH写法的语句访问了2次USER表,如果DEGREE字段又没有索引,性能差异将会非常明显。

 

   当你看到这时,如果很好的理解了上面的内容,我相信你会对WITH语句有了一定的体会。

然而WITH语句能做的还不止这些,下面给大家介绍一下,如何用WITH语句做递归查询。

递归查询的一个典型的例子是对树状结构的表进行查询,考虑如下的情况:

[c-sharp] viewplaincopyprint?

1.论坛首页  

2.--数据库开发  

3.----DB2  

4.------DB2 文章1  

5.--------DB2 文章1 的评论1  

6.--------DB2 文章1 的评论2  

7.------DB2 文章2  

8.----Oracle  

9.--Java技术  

 

以上是一个论坛的典型例子,下面我们新建一个表来存储以上信息。

[c-sharp] viewplaincopyprint?

1.CREATE TABLE BBS  

2.(  

3.PARENTID INTEGER NOT NULL,  

4.ID INTEGER NOT NULL,  

5.NAME VARCHAR(200) NOT NULL---板块、文章、评论等。

  

6.);  

7.insert into bbs (PARENTID,ID,NAME) values   

8.(0,0,'论坛首页'),  

9.(0,1,'数据库开发'),  

10.(1,11,'DB2'),  

11.(11,111,'DB2 文章1'),  

12.(111,1111,'DB2 文章1 的评论1'),  

13.(111,1112,'DB2 文章1 的评论2'),  

14.(11,112,'DB2 文章2'),  

15.(1,12,'Oracle'),  

16.(0,2,'Java技术');  

 

现在万事兼备了,我们开始查询吧。

假设现在让你查询一下‘DB2文章1’的所有评论,有人说,这还不简单,如下这样就可以了。

[c-sharp] viewplaincopyprint?

1.SELECT * FROM BBS WHERE PARENTID=(SELECT ID FROM BBS WHERE NAME='DB

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > 初中教育 > 其它课程

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1