C# DataTable重新排列组合问题

l25926845 2015-11-13 06:25:15
已知在数据库中使用存储过程查询到如下表

注意图中黄色部分的二三四列都是一样的,只是后面送货日期不一样。
要求重新组合成如下样式:

可以看到,蓝色部分的两行合并了。
​绿色部分添加和横向总想的求和

求实现方法或者思路,如果能直接在数据库中使用存储过程完成也可以。

谢谢
...全文
336 14 打赏 收藏 转发到动态 举报
写回复
用AI写文章
14 条回复
切换为时间正序
请发表友善的回复…
发表回复
wfy4422 2015-11-16
  • 打赏
  • 举报
回复
存储过程里用查询拼接字符串的方法,既然你能查出后面不固定列,加group by 应该也不是难事吧?也可以处理Datatable,我做过类似的问题,前面固定的列你可以作为分组依据嘛,可以用linq,也可以用DataTble.ToTable。然后创建另外一个相同架构Datable,即用Clone(),每一组,循环将分组键值及非固定列的值插入去
  • 打赏
  • 举报
回复
select sum(日期), 其他字段 from table group by prochDOc, meterail 用输出参数返回总数,或者返回两个结果集,用dataSet接收 select @total1 = sum(日期), @ total2 = sum(日期)... from table
xdashewan 2015-11-16
  • 打赏
  • 举报
回复
引用 7 楼 l25926845 的回复:
感觉可以在上面代码前先合并?
完全可以在'SELECT * FROM tPo2 PIVOT (SUM(GRQty2) FOR delDate IN ('+@sql+')) a WHERE vendor = '+ '50002903'这句sql外面再套一层用于合并的group by
l25926845 2015-11-16
  • 打赏
  • 举报
回复
引用 6 楼 xdashewan 的回复:
[quote=引用 5 楼 l25926845 的回复:]
根据这个vender查到的结果就是这样的,其中8-10条数据前面都一样,但对应多个送货日期就需要合并了。
不知道这样用您的方法是否还可行?

数据库里直接MAX后面几个字段,group by你应该会的吧[/quote]
上图的结果其实是在SQL中查询出来的

我用了一个存储过程vendor作为变量,将日期作为列名移到了上面
DECLARE @sql VARCHAR(8000)

SELECT @Sql=ISNULL(@sql+',','')+'['+delDate+']' FROM tPo2
WHERE vendor = '50002903'
GROUP BY delDate
ORDER BY delDate DESC

SET @sql='SELECT * FROM tPo2 PIVOT (SUM(GRQty2) FOR delDate IN ('+@sql+')) a WHERE vendor = '+ '50002903'
EXEC (@sql)


原表是这样的


感觉可以在上面代码前先合并?
xdashewan 2015-11-16
  • 打赏
  • 举报
回复
引用 5 楼 l25926845 的回复:
根据这个vender查到的结果就是这样的,其中8-10条数据前面都一样,但对应多个送货日期就需要合并了。 不知道这样用您的方法是否还可行?
数据库里直接MAX后面几个字段,group by你应该会的吧
l25926845 2015-11-16
  • 打赏
  • 举报
回复
引用 3 楼 shingoscar 的回复:
你需要建立一张hashtable,它的key是前4个字段,value是后3个字段
随后遍历数据表,如果某条记录的前4个字段在hashtable中已经存在,就合并。

引用 3 楼 shingoscar 的回复:
你需要建立一张hashtable,它的key是前4个字段,value是后3个字段
随后遍历数据表,如果某条记录的前4个字段在hashtable中已经存在,就合并。

你好,可能我说的有问题。第一张图中只有前几列是固定的,后面的送货日期不一定,有可能是三列也有可能是很多列。
如下图:

根据这个vender查到的结果就是这样的,其中8-10条数据前面都一样,但对应多个送货日期就需要合并了。
不知道这样用您的方法是否还可行?
Poopaye 2015-11-16
  • 打赏
  • 举报
回复
大概写了下 结构
class Key
{
	public string vendor;
	public string purchDoc;
	public string material;
	public string shoText;

	public override int GetHashCode()
	{
		unchecked
		{
			return
				((vendor.GetHashCode() * 33 +
				purchDoc.GetHashCode()) * 33 +
				material.GetHashCode()) * 33 +
				shoText.GetHashCode();
		}
	}

	public override bool Equals(object obj)
	{
		Key other = obj as Key;
		return other != null &&
			other.vendor == this.vendor &&
			other.purchDoc == this.purchDoc &&
			other.material == this.material &&
			other.shoText == this.shoText;
	}
}

class Value
{
	public Dictionary<string, string> data;
}

Dictionary<Key, Value> table; //hashtable
转换过程
DataTable dt; //要操作的数据表

//合并过程
foreach (DataRow row in dt.Rows)
{
	Key key = new Key
	{
		vendor = (string)row["vendor"],
		purchDoc = (string)row["purchDoc"],
		material = (string)row["material"],
		shoText = (string)row["shoText"],
	};

	Value value;

	//查找
	if(!table.TryGetValue(key, out value))
	{
		value = new Value();
		table.Add(key, value);
	}

	//合并value, 5是日期开始的列号
	for (int col = 5; col < dt.Columns.Count; ++col)
	{
		if (!value.data.ContainsKey(dt.Columns[col].ColumnName) && row[col] != null)
			value.data.Add(dt.Columns[col].ColumnName, (string)row[col]);
	}
}

//写入合并后的数据
//这边可以直接写sql
foreach (var item in table)
{
	DataRow newrow = dt.NewRow();
	newrow["vender"] = item.Key.vendor;
	//...........
	foreach (var item2 in item.Value.data)
	{
		newrow[item2.Key] = item2.Value;
	}
	dt.Rows.Add(newrow);
}
xdashewan 2015-11-16
  • 打赏
  • 举报
回复
http://www.cnblogs.com/zhangzt/archive/2010/07/29/1787825.html 注意看行转列的第5个例子和之后的如何分组
l25926845 2015-11-16
  • 打赏
  • 举报
回复
引用 3 楼 shingoscar 的回复:
你需要建立一张hashtable,它的key是前4个字段,value是后3个字段 随后遍历数据表,如果某条记录的前4个字段在hashtable中已经存在,就合并。
你好,如果将前几列作为key的话,那么循环插入时肯定会报错啊 但如果反过来作为value来存储的话,有没有Contains方法
l25926845 2015-11-16
  • 打赏
  • 举报
回复
引用 8 楼 xdashewan 的回复:
[quote=引用 7 楼 l25926845 的回复:] 感觉可以在上面代码前先合并?
完全可以在'SELECT * FROM tPo2 PIVOT (SUM(GRQty2) FOR delDate IN ('+@sql+')) a WHERE vendor = '+ '50002903'这句sql外面再套一层用于合并的group by[/quote] 小弟不才,还请指教具体写法。 写了几种都会报错
l25926845 2015-11-14
  • 打赏
  • 举报
回复
感觉3L的方法可以一试
Poopaye 2015-11-13
  • 打赏
  • 举报
回复
你需要建立一张hashtable,它的key是前4个字段,value是后3个字段 随后遍历数据表,如果某条记录的前4个字段在hashtable中已经存在,就合并。
vivorimage 2015-11-13
  • 打赏
  • 举报
回复
感觉数据已经插入后做这类检查比较费时,对每一条记录都要查询其他记录是否包含与之一样的vendor、purchDoc、material和shoText属性,若表中一共有n条记录,则时间复杂度为n*n。有可能的话还是应该在插入数据时做合并。 合并过程倒直接,若使用DataTable,找到要合并的List<DataRow>,循环加就行了,遇到null就置0。
l25926845 2015-11-13
  • 打赏
  • 举报
回复
当然数据源中为null的地方也得补0
特等奖标准成品论文(Word无水印纯净版) 硬核结构:全文包含完整的摘要、问题重述与分析、模型假设、符号说明、模型建立与求解、灵敏度分析及结论。 即插即用:排版严格遵循官方规范,逻辑严密。拿到手即可作为绝佳的高分参考模板,稍作替换与个性化润色即可极速完稿,彻底解决写论文难的痛点。 双源硬核解题代码(Python与MATLAB双版本) 拒绝假代码:提供底层逻辑清晰、模块化设计的全套可运行源码。 全流程覆盖:涵盖从前期数据清洗预处理,到中期核心数学模型训练,再到后期启发式算法寻优。 傻瓜式运行:代码自带详尽的逐行中文注释,并支持一键生成高质量结果可视化图表,编程小白也能轻松复现与二次开发。 全量数据与结果展示表 所有中间处理数据、模型输出参数以及最终结论,均已精细整理成高质量表格。直观呈现性能评估指标与多模型对比分析,可直接作为论文正文或附件使用,极大提升学术说服力。 独家硬核思路解析 深入浅出剖析出题人意图,详细拆解每一小问的数学本质与底层逻辑,让你不仅知其然更知其所以然。 【四大核心产品优势】 高效实用:所有代码与论文均经过严格测试,确保结果精准无误、完全可复现,省去熬夜试错的时间。 全栈覆盖:从思路分析到跑出结果,再到写出高质量论文,提供一站式全流程资料矩阵。 排版辅助:资料内提供专业的论文排版一键转换工具与官方标准模板,告别格式调整的繁琐。 持续迭代:网盘直发,开赛后资料库将持续滚动更新,所有用户均可免费同步获取最新包。 【适用人群】 想要打破建模瓶颈的参赛队长与主攻手;急需高质量底层代码的编程小白;目标直指特等奖需要高分模板对标的精英团队。
内容概要:本文围绕分布式光伏储能系统的优化配置方法展开,重点基于Matlab代码实现系统建模与仿真分析,涵盖光伏出力不确定性、负荷响应特性及储能充放电行为的综合建模。通过引入改进鲸鱼优化算法、粒子群算法等智能优化算法,解决系统容量配置、能量管理策略与多目标优化调度等问题,旨在提升系统运行效率、降低能耗成本并延长设备使用寿命。研究内容还包括并网与离网模式下的经济性对比、微电网能量优化管理、储能配置方案评估,并提供完整的Matlab代码、数据集及论文撰写模板,支持科研成果的高效产出。; 适合人群:具备电力系统、新能源技术或自动化等相关专业背景,熟悉Matlab/Simulink仿真环境的研究生、科研人员及工程技术人员;特别适用于计划撰写EI/SCI期刊论文的高层次研究人员。; 使用场景及目标:①开展分布式光伏储能系统配置的科研课题与学术论文撰写;②实现微电网能量管理系统的设计与优化调度算法验证;③掌握基于Matlab的多目标优化建模方法,提升在新能源系统规划与运行领域的科研创新能力。; 阅读建议:建议结合提供的Matlab代码与Simulink模型进行动手实践,重点关注优化算法的实现逻辑与参数调优过程,按照文档目录循序渐进学习,构建完整的知识体系,并参考配套论文模板完成研究成果的规范化输出。
监控系统粘贴板和时间截的小软件(开源成世) 梅菉佬--惊屎无人知 2026-06-05 ①使用电脑,经常要用到剪贴板历史的,在windows10中有系统的,但是在ubuntu24中,我找不到,找第三方软件,也是各种不顺限制,有的版本变了就无法使用,又或这样那样的要求,不如干脆自已做一个自用吧。 ②先是,读出粘贴板的历史数据,主要是文本,至于非文本格式,暂不理吧,手机粘贴板历史也是如是。 ③发现,还要加上当前时间截,这个功能很小,但很实用,很多系统没有自带这功能,自已在ubuntu24中加上lazarus用托盘小软件,轻松实现。 ④发现,最好加上常用的难打字,如,塘㙍的㙍字,这个字平时不容易打字出来,事先放入小软件中,到时当成粘贴板历史数据取出来用就行了。又发现,一些五笔中能记忆的字,也可如是,例如,逐,垂,等字。还有一些特殊符号,手机上有,电脑难找但要以显示的,例如,☆★①②③④⑤,还有,一些平时要常查但难记忆的,例如linux的命令,快捷键,也可以加入,相当于托盘记事本。。。。 ⑤还有,在ubuntu中,加入到开机启动中,测试顺利可用。还有,编程时,加入ini文件,可以使用时编写文本,不用每次编译源码。还有,要用分级菜单功能,暂定二级树形菜单,否则太难看了。刚开始用最简单的timer来读取粘贴板数据,后来找到别的API,不用timer了,但是,linux的API比起windows的API,似乎难找得多了。。。 于是,自产自用小工具产生了。实测可用。独乐乐不如众乐乐,于是,开源,念心经,惊S无人知,压惊也,这点小事被卡住,多么不抵。。。 //显式初始化GTK(安全调用)

111,131

社区成员

发帖
与我相关
我的任务
社区描述
.NET技术 C#
社区管理员
  • C#
  • Creator Browser
  • by_封爱
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

让您成为最强悍的C#开发者

试试用AI创作助手写篇文章吧