现在的位置: 首页 > 综合 > 正文

如何从单行提取成多行(TSQL)?

2013年08月24日 ⁄ 综合 ⁄ 共 1434字 ⁄ 字号 评论关闭

Outline:

  • 问题产生背景
  • 交叉连接
  • UNION操作
  • WITH子句
  • 性能比较
  • 抛砖引玉(欢迎指教!)

问题产生背景

从一个仅有数额指标的交易表中创建一个查询,这个查询用于会计日志条目。因此如果一个日志条目应该有存款和借款,但是其他的值应该是相同的,因此我想通过SQL查询从一行提取2行记录。听起来很模糊?我Google之后发现没有任何结果,所以我就写了这篇文章。为了简化,这里举个例子。我有一张表叫做SampleTable,它包含如下记录:

ID

FirstName

LastName

1

Anna

Gates

2

John

Doe

3

Joe

Bloggs

4

Raj

Kumar

现在你的任务是为每个记录创建3行,因此它将显示如下:

ID

FirstName

LastName

ItemNumber

ItemDescription

1

Anna

Gates

1

Item 1

1

Anna

Gates

2

Item 2

1

Anna

Gates

3

Item 3

2

John

Doe

1

Item 1

2

John

Doe

2

Item 2

2

John

Doe

3

Item 3

3

Joe

Bloggs

1

Item 1

3

Joe

Bloggs

2

Item 2

3

Joe

Bloggs

3

Item 3

4

Raj

Kumar

1

Item 1

4

Raj

Kumar

2

Item 2

4

Raj

Kumar

3

Item 3

现在,你如何实现它呢?有许多种方法,但是我们要找出哪个方法查询效率最高。有下面是那种方法:

  • 交叉连接
  • UNION查询
  • WITH查询

1、交叉连接

为了激活大家对交叉连接沉睡的记忆,首先介绍下什么是交叉连接。所谓交叉连接,就是两个表的笛卡尔积的另一称谓。交叉连接为将第一张表的每一行与第二张表的每一行组合产生一新的元组。设两张表R、S分别有k1、k2条记录,每条记录的列数分别为m、n,则交叉连接的结果元组数为k1*k2,每个元组的列数为m+n(前面m列是R的,后面n列是S的)。当然这是在没有where条件的情况下,如果加了where添加可能会过滤掉一部分不符合条件的记录。

所以上面的结果可以看成下面两张表的交叉连接产生的:

image 因此可以用如下SQL语句:

本来是打算对表2构建一张临时表,但考虑到SQL Server与Oracle构建临时表是有差异的,考虑到这个我就用上面这种方式(UNION ALL,集合查询)。

2、UNION操作

UNION是集合操作中的一种,SELECT语句的查询结果是元组的集合,所以多个SELECT语句的结果可进行集合操作。集合操作主要包括并操作UNION、交操作INTERSECT和差操作EXCEPT。注意,参加集合操作的各查询结果的列数必须相同;对应项的数据类型也必须相同。

下面是实现代码:

抱歉!评论已关闭.