SQL SERVER 实现多行转多列
有这样一个需求,一个表单主表,一个扩展列表,查询的时候要把扩展列表中的多行转成主表多列。
比如
dt_zhubiao [主表]
| id | type | title |
| 1 | 1 | 表单1-1 |
| 2 | 1 | 表单1-2 |
| 3 | 2 | 表单2-1 |
| 4 | 2 | 表单2-2 |
dt_kuozhanbiao [扩展表]
| id | formid | name | title | value |
| 1 | 1 | ext_a | 工龄 | 18 |
| 2 | 1 | ext_b | 职称 | 副级 |
| 3 | 2 | ext_a | 工龄 | 20 |
| 4 | 2 | ext_b | 职称 | 正级 |
| 5 | 3 | ext_2a | 字段1 | 值1 |
| 6 | 3 | ext_2b | 字段2 | 值2 |
| 7 | 3 | ext_2c | 字段3 | 值3 |
| 8 | 4 | ext_2a | 字段1 | 值21 |
| 9 | 4 | ext_2b | 字段2 | 值22 |
| 10 | 4 | ext_2c | 字段3 | 值23 |
查询时,会根据dt_zhubiao表的type来查询,type字段一样时,dt_kuozhanbiao表条数和name都会一致,value不一致。
想要的结果如下:
查询type=1时,select * from dt_zhubiao where type = 1 ...
| id | type | title | ext_a | ext_b |
| 1 | 1 | 表单1-1 | 18 | 副级 |
| 2 | 1 | 表单1-2 | 20 | 正级 |
查询type=2时,select * from dt_zhubiao where type =2 ...
| id | type | title | ext_2a | ext_2b | ext_2c |
| 3 | 2 | 表单2-1 | 值1 | 值2 | 值3 |
| 4 | 2 | 表单2-2 | 值21 | 值22 | 值23 |
那么问题来了,基于 select * from dt_zhubiao where type = ? 基础sql语句,如何生成这种查询结果 ?
这个问题应该多用于动态表单,之前自己尝试过join 、union去解决,都总差那么点意思。 -:)
后面去多个论坛发帖挨个问了个遍,终于寻到解决办法了。SQL 代码如下:
--测试数据
if not object_id(N'Tempdb..#主表') is null
drop table #主表
Go
Create table #主表([id] int,[type] int,[title] nvarchar(25))
Insert #主表
select 1,1,N'表单1-1' union all
select 2,1,N'表单1-2' union all
select 3,2,N'表单2-1' union all
select 4,2,N'表单2-2'
GO
if not object_id(N'Tempdb..#扩展表') is null
drop table #扩展表
Go
Create table #扩展表([id] int,[formid] int,[name] nvarchar(26),[title] nvarchar(23),[value] nvarchar(22))
Insert #扩展表
select 1,1,N'ext_a',N'工龄',N'' union all
select 2,1,N'ext_b',N'职称',N'副级' union all
select 3,2,N'ext_1',N'工龄',N'' union all
select 4,2,N'ext_b',N'职称',N'正级' union all
select 5,3,N'ext_2a',N'字段1',N'值1' union all
select 6,3,N'ext_2b',N'字段2',N'值2' union all
select 7,3,N'ext_2c',N'字段3',N'值3'union all
select 8,4,N'ext_2a',N'字段1',N'值1' union all
select 9,4,N'ext_2b',N'字段2',N'值2' union all
select 10,4,N'ext_2c',N'字段3',N'值3'
Go
--测试数据结束
DECLARE @sql VARCHAR(MAX)
SET @sql = 'select #主表.id,#主表.type,#主表.title'
SELECT @sql = @sql + ',max(case name when ''' + name
+ ''' then [value] else null end)[' + name + ']'
FROM ( Select DISTINCT name from #扩展表 JOIN #主表 ON formid IN (SELECT id FROM #主表 WHERE type=2)
) a
SET @sql = @sql
+ ' from #扩展表 JOIN #主表 ON formid =#主表.id WHERE type=2 group by #主表.id,#主表.type,#主表.title'
EXEC(@sql)
至此已经解决了我的问题,但是对于实际的项目运用还是缺少点什么,比如 分页、条件筛选。那么得在此基础上稍微修改一下,这个简单我自己会做了 -:) 。
DECLARE @sql VARCHAR(MAX)
SET @sql = 'with tb as (select ROW_NUMBER() OVER(Order by #主表.id ) AS rindex,#主表.id,#主表.type,#主表.title'
SELECT @sql = @sql + ',max(case name when ''' + name
+ ''' then [value] else null end)[' + name + ']'
FROM ( Select DISTINCT name from #扩展表 JOIN #主表 ON formid IN (SELECT id FROM #主表 WHERE type=2)
) a
SET @sql = @sql
+ ' from #扩展表 JOIN #主表 ON formid =#主表.id WHERE type=2 group by #主表.id,#主表.type,#主表.title); select * from tb where ext_2b =''值1'' and rindex between 1 and 10 ' --拼接with 及查询条件
EXEC(@sql)
到这自我感觉应该差不多了,再改改做成个存储过程应该可以用了,但是吧,想着用程序也去实现一遍,看看哪种实行起来更方便 -:)
程序实现两种思路,一种组装table:先结合分页、条件筛选等查出需要的主表数据集得到一个datatable,然后给datatable动态去添加对应列,然后循环去赋值(这种情况就无法实现针对扩展字段进行排序)。
DataTable tb = DbHelperSQL.Query("with tb as (select row_number over(order by #主表.id) as rindex,* from #主表 where id in (select formid from #扩展表 where value ='值2') adn type=2 );select * from tb where rindex between 1 and 10").Tables[];
var ar = new System.Collections.ArrayList();
foreach (DataRow crow in DbHelperSQL.Query("select name from #扩展表 where formid in (select id from #主表 where type = 2)").Tables[].Rows) {
ar.Add(crow[]);
tb.Columns.Add(crow[].ToString(), typeof(string));
}
//这里可以一次性加载tb数据集中包含的所有#扩展表数据,然后在内存中进行操作赋值,如下循环查询数据库赋值只是为了写的方便
for (var i =0; i <tb.Rows.Count;i++) {
var formid = tb.Rows[i]["id"];
foreach (var ari in ar) {
tb.Rows[i][ari.ToString()] = DbHelperSQL.GetSingle(string.Format("select value from #临时表 where formid ={0} and name ='{1}'",formid,ari));
}
}
return tb;
第二种就是组装SQL语句啦,但是不在程序中组装之前的@sql字符串,但是还是参考之前的SQL代码思路,然后转成程序代码思路。在之前SQL实现的代码中,最后的exec(@sql)前一行加上打印@sql的语句,就会得到最后执行的sql语句
select #主表.id,#主表.type,#主表.title
,max(case name when 'ext_2a' then [value] else null end)[ext_2a]
,max(case name when 'ext_2b' then [value] else null end)[ext_2b]
,max(case name when 'ext_2c' then [value] else null end)[ext_2c]
from #扩展表 JOIN #主表 ON formid =#主表.id WHERE type=2 group by #主表.id,#主表.type,#主表.title
基于这个SQL语句用程序去拼接最终的SQL语句执行(包括分页、条件筛选)。
string sql=" with tb as (select row_number over(order by #主表.id) rindex, #主表.id,#主表.type,#主表.title";
foreach (DataRow crow in DbHelperSQL.Query("select name from #扩展表 where formid in (select id from #主表 where type = 2) group by name").Tables[].Rows) {
sql +=",max(case name when '"+crow["name"]+"' then [value] else null end) "+crow["name"];
}
sql+=" from #扩展表 join #主表 on formid = #主表.id where type = 2 group by #主表.id,#主表.type,#主表.title); select * from tb where ext_2a ='值1' and rindex between 1 and 10 ";
return DbHelperSQL.Query(sql).Tables[];
//此处单纯的SQL语句拼接,也可以再次优化实现动态参数化。
这种情况是可以实现所有的列都能进行排序。 目前来讲,还只是去实现这个功能,还没有考虑性能优化之类的。
论坛原咨询帖:https://bbs.csdn.net/topics/392999794 。 记于此以作备份,好记性不如烂笔头。 -:)
SQL SERVER 实现多行转多列的更多相关文章
- Sql server 中将数据行转列列转行(二)
老规矩,先弄一波测试数据,数据填充代码没有什么意义,先折叠起来: /* 第一步:创建临时表结构 */ CREATE TABLE #Student --创建临时表 ( StuName ), --学生名称 ...
- 在SQL Server 2014里可更新的列存储索引 (Updateable Column Store Indexes)
传统的关系数据库服务引擎往往并不是对超大量数据进行分析计算的最佳平台,为此,SQL Server中开发了分析服务引擎去对大笔数据进行分析计算.当然,对于数据的存放平台SQL Server数据库引擎而言 ...
- 通过DBCC Page查看在SQL Server中哪行数据被锁住了?
原文:通过DBCC Page查看在SQL Server中哪行数据被锁住了? 如何查看被锁的是哪行数据?通过dbcc page可以. 要想明白这个问题: 首先,需要模拟阻塞问题,这里直接模拟了阻塞问题的 ...
- SQL Server删除重复行的6个方法
SQL Server删除重复行是我们最常见的操作之一,下面就为您介绍六种适合不同情况的SQL Server删除重复行的方法,供您参考. 1.如果有ID字段,就是具有唯一性的字段 delect ta ...
- SQL Server获取指定行的数据
SQL Server获取指定行(如第二行)的数据 --SQL Server获取指定行(如第二行)的数据-- --法一(对象法)-- select * from ( select * , numbe ...
- 向SQL Server 现有表中添加新列并添加描述.
注: sql server 2005 及以上支持. 版本估计是不支持(工作环境2005,2008). 工作需要, 需要向SQL Server 现有表中添加新列并添加描述. 从而有个如下存储过程. (先 ...
- SQL Server 中 ROWLOCK 行级锁
一.ROWLOCK的使用 1.ROWLOCK行级锁确保,在用户取得被更新的行,到该行进行更新,这段时间内不被其它用户所修改.因而行级锁即可保证数据的一致性,又能提高数据操作的并发性. 2.ROWLOC ...
- SQL SERVER pivot(行转列),unpivot(列转行)
[pivot]行转列:多行变一列 假设学生成绩表Score1 Name Subject Score 小张 语文 88 小花 数学 89 小张 数学 90 Name 语文 数学 小花 null 89 小 ...
- 在一个SQL Server表中的多个列找出最大值
在一个SQL Server表中一行的多个列找出最大值 有时候我们需要从多个相同的列里(这些列的数据类型相同)找出最大的那个值,并显示 这里给出一个例子 IF (OBJECT_ID('tempdb..# ...
随机推荐
- [报错解决] k8s 删除pv一直处于terminating 两种解决方法
第一种 直接到etcd中删除 1.将所有的etcd中的key值取到一个keys.yam里面,便于查询 ETCDCTL_API=3 etcdctl get "" --from-key ...
- 静态blog的免费托管部署、加域名与搜索优化(SEO)
本文通过MetaWeblog自动发布,原文及更新链接:https://extendswind.top/posts/technical/hugo_blog_host_and_seo 给博客加个域名准备长 ...
- vue从一个组件跳转到另一个组件页面router-link的试用
需求从helloworld.vue页面跳到good.vue页面 1.helloworld.vue页面代码 <template> <div class="hello" ...
- Jira5.2.8 安装
第一步. 安装jira 1. 运行安装包,选择自定义安装 2. 最好使用netstat -ano检查一下端口是否被占用,如果占用就选其他的靓号吧~ 3. 选择集成数据库 输入应用的名称 第二步. 破解 ...
- bs4 string与text的区别
用python写爬虫时,BeautifulSoup真是解析html,快速获取所需数据的神器. 这个美味汤使唤起来,屡试不爽. 在用find()方法找到特定的tag后,想获取里面的文本,可以用.text ...
- java 抽象类为什么不能被实例化?
我把CSDN论坛里面的一个帖子内容list到下面,自己看着理解,东家一言,西家一语,杂合起来,基本上也就理解了java中的抽象类为什么不能被实例化了. 因篇幅有限,只能罗列部分留言 以下内容不分先后顺 ...
- PHP网络服务
[Socket] socket_create 用于创建一个Socket socket_bind 用于将IP地址和端口绑定到socket_create 函数所创建的句柄中. socket_listen ...
- connections java.net.BindException: Address already in use_解决方案
一.问题描述 在Linux服务器(CentOS7系统)中配置并启动JMeter远程监控服务器资源所需的ServerAgent目录下的 startAgent.sh 文件时,系统出现异常提示,如 [roo ...
- Qt编写控件属性设计器2-拖曳控件
一.前言 上一篇文章把插件加载好了,并且把插件中的所有控件都显示到了列表框中,这次要做的就是实现拖曳控件的功能,用户选择一个控件拖曳到画布上,松开,在松开位置处自动实例化该控件,这个需要用到dropE ...
- 在node.js中使用Set
var set = new Set(); set.add(1); console.log("test1 : " + set.has(1) + " ; " + s ...