【C#】Excel导出合并行和列并动态加载行与列
简单的Excel导出比较好做,只要设置表头,循环在表格中赋值添加数据即可,但是如果表头是不固定的,并且个数是不确定的,这就需要根据查询出数据的特点来添加导出了。
导出效果图:
如上图所示,商品的个数是不确定的,时间的月份个数也是不确定的,所以简单的通过模板是不可以的。并且数据库中查询出的信息是每个商品不同时间的信息,所以查询出的数据相同时间的可能有多条,一个商品在不同的时间分布,所以也可以是多条。
代码部分:
- public int DataTableToExcelByMProduct(DataTable dt_model, string sheetName)
- {
- workbook = new HSSFWorkbook();
- ISheet sheet = workbook.CreateSheet(sheetName);
- IRow row = null;
- ICell cell = null;
- //样式
- ICellStyle style = workbook.CreateCellStyle();
- style.Alignment = HorizontalAlignment.CENTER;//设置单元格的样式:水平对齐居中
- style.VerticalAlignment = VerticalAlignment.CENTER;//设置单元格样式:垂直对齐居中
- IFont font = workbook.CreateFont();//新建一个字体样式对象
- font.Boldweight = short.MaxValue;
- style.SetFont(font);
- DateTime nowTime = DateTime.Now;
- //商品编码数量
- var dtgroup=(from p in dt_model.AsEnumerable()
- group p by new {
- RealName=p.Field<string>("RealName"),
- Name=p.Field<string>("Name")
- }into g
- select new {
- RealName=g.Key.RealName,
- Name=g.Key.Name,
- counts=g.Count()
- }).ToList();
- //数据表头,时间
- row = sheet.CreateRow(0);
- cell = row.CreateCell(0);
- cell.SetCellValue("时间");
- cell.CellStyle = style;
- //CellRangeAddress四个参数:起始行、结束行、起始列、结束列
- sheet.AddMergedRegion(new CellRangeAddress(0, 2, 0, 0));
- //进出口岸 1,1,2,口岸数*2
- cell = row.CreateCell(1);
- cell.SetCellValue(“进出口商品编码”);
- cell.CellStyle = style;
- //CellRangeAddress四个参数:起始行、结束行、起始列、结束列
- sheet.AddMergedRegion(new CellRangeAddress(0, 0, 1, 2*dtgroup.Count));
- //商品拼接
- row = sheet.CreateRow(1);
- row = sheet.CreateRow(2);
- for( int c=0;c<dtgroup.Count;c++)
- {
- //创建进出口岸单元格,开始
- cell = row.CreateCell(2*c+1);
- cell.SetCellValue(dtgroup[c].RealName);
- cell.CellStyle = style;
- //CellRangeAddress四个参数:起始行、结束行、起始列、结束列
- sheet.AddMergedRegion(new CellRangeAddress(1, 1, 2*c+1, 2*c+2));
- //创建进口单元格
- //创建一列
- cell=row.CreateCell(2*c+1);
- cell.SetCellValue("进口");
- cell=row.CreateCell(2*c+2);
- cell.SetCellValue("出口");
- }
- var AllYearCount=(from p in dt_model.AsEnumerable()
- group p by new {Year_Month=p.Field<string>("Year_Month")} into m
- select new
- {
- YearMonth =m.Key.Year_Month,
- AllYearCount=m.Count()
- }).ToList();
- //年份
- for (int i=0;i < AllYearCount.Count;i++)
- {
- row=sheet.CreateRow(i+3);
- cell=row.CreateCell(0);
- string YearMonth=AllYearCount[i].YearMonth;
- int month =SafeConvert.ToInt16(YearMonth.Substring(4,2));
- cell.SetCellValue("1-"+month+"月");
- //贸易额对碰情况
- for (int j=0;j<dtgroup.Count;j++)
- {
- var items=dt_model.AsEnumerable().Where(a=>a.Field<string>("Year_Month")==AllYearCount[i].YearMonth && a.Field<string>("Name")==dtgroup[j].Name).ToList();
- if(items.Count>0)
- {
- cell=row.CreateCell(2*j+1);
- cell.SetCellValue(items[0].Field<string>("Export_Desn"));
- cell=row.CreateCell(2*j+2);
- cell.SetCellValue(items[0].Field<string>("Import_Desn"));
- }
- else
- {
- cell=row.CreateCell(2*j+1);
- cell.SetCellValue("");
- cell=row.CreateCell(2*j+2);
- cell.SetCellValue("");
- }
- }
- }
- using(FileStream fsm=File.Open(fileName,FileMode.OpenOrCreate,FileAccess.ReadWrite))
- {
- workbook.Write(fsm);
- fsm.Close();
- }
- return 1;
- }
其实代码的核心部分就是创建行与列并且为表格赋值,如果已经创建了行就不用创建了,就像这个例子中的时间单元格已经创建了一次行,这样“进出口商品编码”就不用再次创建行了。但是创建了行必须要创建列,列是在行的基础上创建的,所以即使上一行已经创建了列,下一行还是需要重新创建的。
小结:
这个方法传入的是datatable和表格名称,如果我们返回的数据不是直接输出的需要做一些处理,我们可以采用给datatable增加字段的方法,将我们想要的结果存储到新加的字段中。
导出的思想是一样的,都是要循环行和列,在表格中赋值,不同的是从哪里开始赋,把不同的地方解决,导出也一样easy!
【C#】Excel导出合并行和列并动态加载行与列的更多相关文章
- 只用css实现“每列四行,加载完一列后数据自动填充到下一列”的效果
只用css实现“每列四行,加载完一列后数据自动填充到下一列”的效果.这个题目用图表示如下: 如果将题目换成“只用css实现每行四列,加载完一行后数据自动填充到下一行”,那这个问题就简单多了,相信大家都 ...
- Java获取后台数据,动态生成多行多列复选框
本例目标: 获取后台数据集合,将集合的某个字段,比如:姓名,以复选框形式显示在HTML页面 应用场景: 获取数据库的人员姓名,将其显示在页面,供多项选择 效果如下: 一.后台 查询数据库,返回List ...
- js动态加载数据并合并单元格
js动态加载数据合并单元格, 代码如下所示,可复制直接运行: <!DOCTYPE HTML> <html lang="en-US"> <head> ...
- jquery easyui datagrid 动态 加载列
实现方式: 首先根据输入的sql语句获得相关的列名称返回给前台,然后在datagrid中动态加载列,接着根据查询条件(包括sql语句)获取相关的记录返回给前台用于填充datagrid.从而实现类似or ...
- Excel催化剂开源第7波-VSTO开发中Ribbon动态加载菜单
在VS开发环境中,特别是VSTO的开发,微软已经现成地给开发者准备了设计器模式的功能区开发,相对传统的VBA.ExcelDna和其他方式的COM加载项开发来说,不需要手写xml功能区,直接类似拖拉窗体 ...
- c# 创建Excel com加载项Ribbon动态加载工作簿和工作表
使用 VSTO 创建外接程序,Gallery控件动态加载工作簿名称 代码如下: 加载工作簿名称: private void Gallery1_ItemsLoading(object sender, R ...
- 【datagrid】动态加载列 2016-01-03 16:32 2013人阅读 评论(19) 收藏
之前我们的项目在前台显示只需要把数据从数据库读出来进行显示就可以,datagrid的表头字段都是写死的,把数据往表里一扔,就基本没什么事儿了,结果客户前几天要求,其中一个字段不能是死的,应该是有多少项 ...
- (转)高性能JavaScript:加载和运行(动态加载JS代码)
浏览器是如何加载JS的 当浏览器遇到一个<script>标签时,浏览器首先根据标签src属性下载JavaScript代码,然后运行JavaScript代码,继而继续解析和翻译页面.如果需要 ...
- AMD和RequireJS初识----优化Web应用前端(按需动态加载JS)
RequireJS是一个非常小巧的JavaScript模块载入框架,是AMD规范最好的实现者之一.最新版本的RequireJS压缩后只有14K,堪称非常轻量.它还同时可以和其他的框架协同工作,使用Re ...
随机推荐
- Orchard模块开发全接触7:订单与支付之Event Bus
在这部分,我们要完成的工作有: 1:将购物车内的商品变成真正的订单: 2:理解 父子及一对多关系: 3:写一个针对 Event Bus 的扩展点: 4:实现一个针对该扩展点的模拟的 支付服务: 一:创 ...
- [转]PHP中替换换行符
FROM :http://www.cnblogs.com/siqi/archive/2012/10/12/2720713.html //php 有三种方法来解决 //1.使用str_replace 来 ...
- [转]使用HackCube-Special分析胎压传感器信号
胎压无线传感器安全检测 我们团队之前也有用USRP和GNUradio对其他的胎压设备进行的安全检测,我不使用这套环境的原因是软件无线电的设备和笔记本已经算体积不小的一套设备,通常测试环境都在户外,在这 ...
- Valid Number leetcode java
题目: Validate if a given string is numeric. Some examples: "0" => true " 0.1 " ...
- 详解Vue中的nextTick
Vue中的nextTick涉及到Vue中DOM的异步更新,感觉很有意思,特意了解了一下.其中关于nextTick的源码涉及到不少知识,很多不太理解,暂且根据自己的一些感悟介绍下nextTick. 一. ...
- Android -- Property Animation
3.0以前,android支持两种动画模式,tween animation,frame animation,在android3.0中又引入了一个新的动画系统:property animation,这三 ...
- 无法从“void (__thiscall CtestDlg::* )(void)”转换为“LRESULT (__thiscall CWnd::* )(WPARAM,LPARAM)”
按照孙鑫的教程添加自定义消息时,如果是VC6.0开发环境,也许没有什么 问题,但在VS2008中编译会报错的 无法从"void (__thiscall CtestDlg::* )(voi ...
- 【Python】torrentParser1.03
#------------------------------------------------------------------------------------ # torrentParse ...
- 浅谈压缩感知(九):正交匹配追踪算法OMP
主要内容: OMP算法介绍 OMP的MATLAB实现 OMP中的数学知识 一.OMP算法介绍 来源:http://blog.csdn.net/scucj/article/details/7467955 ...
- oracle v$sqlarea 分析SQL语句使用资源情况 确认是否绑定变量
-如何确定系统中是否存在绑定变量的情况:首先创建一个表,用于存放整理过得数据:create table t1 as select sql_text from v$sqlarea;----V$SQLAR ...