C#项目自定义汇总导出Excel功能

项目中经常需要导出excel,有时还有汇总需求。但导出方式都一样,而汇总方式不同。可以参考以下方式,使用标准的导出方式,和自定义的汇总方式。

1.首先NuGet引用NPOI version 2.3.0

2.导出表格源码

/// <summary>
        /// 【汇总的外置方法】func第一个参数是上次的行数据,第二个参数是当前行数据,第三个参数是Sheet中的当前行,第四个参数是否是最后一行数据,如果增加了新行,则必须返回true
        /// </summary>
        /// <param name="dt"></param>
        /// <param name="func"></param>
        /// <returns></returns>
        public void ExportDataToExcel(DataTable dt,  string FileName, Func<DataRow, DataRow, IRow,Boolean, Boolean> func)
        {
            HSSFWorkbook hssfworkbook = new HSSFWorkbook();
            ICellStyle cellstyle = hssfworkbook.CreateCellStyle();//设置垂直居中格式
            cellstyle.VerticalAlignment = VerticalAlignment.Top;    //垂直居中
            cellstyle.Alignment = HorizontalAlignment.Center;       //居中

            #region 初始化Sheets
            List<ISheet> Sheets = new List<ISheet>();//Sheets集合
            Action addSheet=()=>{
                ISheet sheet = hssfworkbook.CreateSheet();
                IRow row = sheet.CreateRow(0);
                //表头
                for (int hi = 0; hi < dt.Columns.Count; hi++)
                {
                    ICell cell = row.CreateCell(hi);
                    cell.SetCellValue(dt.Columns[hi].ColumnName);
                    cell.CellStyle = cellstyle;
                }
                Sheets.Add(sheet);
            };
            
            #endregion
            int sheetIndex = 0;
            int rowcount = -1;
            DataRow lastIRow = null;
            try
            {
                addSheet();
                for (int i = 0; i < dt.Rows.Count; i++)
                {
                    rowcount++;
                    if (rowcount > 65535 * (sheetIndex+1))
                    {
                        addSheet();
                        ++sheetIndex;
                    }
                    int currentI = rowcount - (sheetIndex * 65535);
                    IRow row1 = Sheets.ToArray()[sheetIndex].CreateRow(currentI + 1);
                    if (lastIRow==null)
                    {
                        lastIRow = dt.Rows[i];
                    }
                    DataRow currentDataRow = dt.Rows[i];
                    Boolean isend =(i==dt.Rows.Count-1);
                    if (!isend)
                    {
                        var isaddCount = func(lastIRow, currentDataRow, row1, isend);
                        if (isaddCount)
                        {
                            rowcount++;
                            currentI = rowcount - (sheetIndex * 65535);
                            row1 = Sheets.ToArray()[sheetIndex].CreateRow(currentI + 1);
                        }
                    }
                    for (int j = 0; j < dt.Columns.Count; j++)
                    {
                        ICell cell = row1.CreateCell(j);
                       
                        string colName = dt.Columns[j].ColumnName;
                        if (colName.IndexOf("监管码") > 0 || colName.IndexOf("产品码") > 0 || colName.IndexOf("中奖用户") > 0)
                        {

                            cell.SetCellValue("'" + dt.Rows[i][j].ToString());
                            cell.CellStyle = cellstyle;
                        }
                        else
                        {
                            cell.SetCellValue(dt.Rows[i][j].ToString());
                            cell.CellStyle = cellstyle;
                        }

                    }
                    if (isend)
                    {
                       func(lastIRow, currentDataRow, row1, isend);                        
                    }
                    
                    lastIRow = currentDataRow;
                }
                //转为字节数组
                MemoryStream stream = new MemoryStream();
                hssfworkbook.Write(stream);

                stream.Position = 0;
                StreamReader rd = new StreamReader(stream, System.Text.Encoding.BigEndianUnicode);
                StringBuilder sb = new StringBuilder();
                sb.Append(rd.ReadToEnd());
                stream.Close();
                hssfworkbook.Close();

                StringWriter sw = new StringWriter();
                sw.Write(sb.ToString());
                HttpContext.Current.Response.AppendHeader("Content-Disposition", "attachment;filename=" + HttpUtility.UrlEncode(FileName + ".xls", System.Text.Encoding.UTF8));
                HttpContext.Current.Response.ContentType = "application/ms-excel";
                System.Web.HttpContext.Current.Response.ContentEncoding = System.Text.Encoding.BigEndianUnicode;
                System.Web.HttpContext.Current.Response.Write(sw);
                sw.Close();
                System.Web.HttpContext.Current.Response.End();
               

            }
            catch (Exception ex)
            {
                hssfworkbook.Close();
                Acexe.Common.Utils.Log("error", String.Format("ExportDataToExcel {0}{1}{2}", ex.Message, System.Environment.NewLine, ex.StackTrace));
            }
        }

3.自定义参考源码

 

Func<DataRow, DataRow, IRow,Boolean, Boolean> func = (a, b, c,d) =>
            {
                ICellStyle cellstyle = c.Sheet.Workbook.CreateCellStyle();//设置垂直居中格式
                cellstyle.VerticalAlignment = VerticalAlignment.Top;    //垂直居中
                cellstyle.Alignment = HorizontalAlignment.Right;       //居中
               
                if(d)
                {
                    if (Decimal.TryParse(b["红包金额"].ToString(), out recordMoney))
                    {
                        moneyCount += recordMoney;
                    }
                    ICell cell = c.Sheet.CreateRow(c.RowNum+1).CreateCell(0);
                    cell.SetCellValue("UserId:" + a["UserId"].ToString() + " 汇总: " + moneyCount.ToString("0.00"));
                    cell.CellStyle = cellstyle;
                    c.Sheet.AddMergedRegion(new CellRangeAddress(c.RowNum+1, c.RowNum+1, 0, a.Table.Columns.Count - 1));
                    return true;
                }
                else
                {
                    if (!(a["UserId"].ToString().Equals(b["UserId"].ToString())))
                    {
                        ICell cell = c.CreateCell(0);
                        cell.SetCellValue("UserId:" + a["UserId"].ToString() + " 汇总: " + moneyCount.ToString("0.00"));
                        cell.CellStyle = cellstyle;
                        c.Sheet.AddMergedRegion(new CellRangeAddress(c.RowNum, c.RowNum, 0, a.Table.Columns.Count - 1));
                        if (Decimal.TryParse(b["红包金额"].ToString(), out recordMoney))
                        {
                            moneyCount = recordMoney;
                        }
                        return true;
                    }
                    else
                    {
                        if (Decimal.TryParse(b["红包金额"].ToString(), out recordMoney))
                        {
                            moneyCount += recordMoney;
                        }
                        return false;
                    }
                }
            };

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
本DataGridView打印控件和.NET打印控件5.7版2014年11月2日修改完成,完全免费,在.NET2.0及以上环境下都可以使用(VB打印、C#打印都是可以的),有帮助文档与使用实例。 与上一版相比,控件5.7版的主要更改如下: 1、重写DataGridView导出Excel的代码,5.7版控件使用开源的NPOI库导出Excel,完全不依赖Office,不会再有因为没装Office或Office安装有问题而导出失败的情况,而且导出速度非常快,支持大于65536条记录的导出(自动分成多个工作表),效果非常好。导出Excel的接口未变,因此您不需要修改任何代码,只需替换控件即可; 2、SimpleReport打印组件添加了导出Excel功能,该功能在打印预览界面的按钮中; 3、多表头组件MulHeaderDataGridView添加了从Excel中导入数据(ImportFromExcel函数)及复制(Copy函数)与粘贴(Paste函数)的功能。其中导入Excel功能使用开源的NPOI组件实现,不依赖Office。 4、其他一些完善,比如解决了导出Excel时强制换行不自动显示,而是要双击才显示问题;导出Excel时,图片能按单元格大小导出。 本控件特色: 1、强大的DataGridView打印功能,不仅可以以多种形式(普通打印、分栏打印、跨页打印、工资条打印)打印DGV表格,基本上能完全按DGV控件本身设置的格式如字体、字号、背景颜色、前景颜色、单元格对齐方式等打印出来,文字图像都可以打印,而且是完全根据表格当前的显示顺序进行打印的,基本上做到了所见即所得的打印。 2、报表设计功能。报表模板设计组件EasyReport可以设计普通报表、分组报表、套打模板等,以DataGridView为数据源。控件的位置以毫米为计量单位,定位准确,很适合套打单据设计。 3、强大的图表打印功能。5.2版控件新增了一个Chartlet的组件,使用非常方便,可以生成柱形图、饼图、折线图等多种图形,而且可以设置2D或3D效果,既可以在打印控件中打印出来,也可以在Graphics对象中显示。 4、分组汇总打印DataGridVeiw功能,每组还可以自动换新页打印,还可以自动增加行号。 5、强大的文本打印输出功能,控件提供多个文本打印重载函数,打印文本时,如果需要,控件会自动换行和换页打印输出。还增加了以指定行间距及字符间距打印文本的功能,可以用固定行距,也可以用单倍或多倍行距打印文本。 6、强大的绘图功能,基本上.NET的GDI+的绘图函数(如直线、矩形、路径、多边形、曲线等)都有,只有个别函数的名称有点区别。 7、支持同一文档多种版面格式打印(类似于Word中的节的功能):对同一份文档,不同的页面可以设置不同的格式(纸张大小、纸张方向、页边距),只需要在新增一页时在NewPage方法中指定要使用的页面格式即可,使用非常简单。 8、报表文件保存功能。本控件允许将当前打印预览的内容保存为报表文件,以后使用本控件重新打开该报表文件即可重现原来保存报表时的打印内容。 9、Excel导出功能,可以将DataGridView导出Excel文件,5.7版控件使用开源的NPOI导出Excel,速度非常快,效果非常好,5.4版还增加了合并单元格的导出功能。 10、打印DataGridView时的打印方案保存与读取功能。可以将当前打印参数保存为打印方案文件,或者从保存的打印方案文件中读取打印参数。 11、水印打印功能。根据需要,可以在页面中打印或不打印以半透明空心文字打印水印。 12、强大的容器控件打印功能(DrawPanel函数)。借助该函数,您只需要在您的容器控件中设计好要打印的内容及打印内容的相对位置,控件轻松帮你打印出来(如果超过一页,控件会自动换页续打)。 13、特殊文字效果打印功能。控件具有打印浮雕文字、阴影文字、空心文字、块文字的功能。 14、页眉页脚中既可打印文字,也可打印图像,或者即打印图像又打印输出文字。 15、图像与图标打印输出功能。 16、多表头(跨行跨列的复杂表头)打印功能,多表头组件支持多表头显示与打印、单元格内容的合并显示、打印与导出。 17、自定义纸张支持功能。 18、纸张背景图片设置打印功能。 19、.NET4.0支持功能(是单独的一个文件)。 20、直接打印窗口中的TreeView控件功能。 21、打印窗口中的ListView功能。 22、RichTextBox控件的RTF文本打印功能。 23、斜线表头打印功能(5.4版新增)。 24、二维码打印功能(5.5版本增加)。 25、5.6版新增的SimpleReport组件允许您在一个方案文件中管理多个打印方案,在打印预览时能自由在各个打印方案之间切换。 26、5.7版控件增加了使用开源的NPOI从Excel文件中导入数据到DataGridView的功能,以及DataGridView的复制与粘贴功能。 我将持续改进该控件,并将不断推出控件的新版本,要查看或下载控件的升级版本,请登陆网站:http://myyouping.download.csdn.net/ 。具体使用方法请参见帮助文件与实例文件,如有疑问或好的建议,请与我联系: 邮箱:[email protected] Q Q:479781502
具体内容请参考我的BLOG:http://blog.csdn.net/smallwhiteyt/archive/2009/11/08/4784771.aspx 如果你耐心仔细看完本文,相信以后再遇到导出EXCLE操作的时候你会很顺手觉得SO EASY,主要给新手朋友们看的,老鸟可以直接飘过了,花了一晚上的时间写的很辛苦,如果觉得对你有帮助烦请留言支持一下,我会写更多基础的原创内容来回报大家。 C#导出数据到EXCEL表格是个老生常谈的问题了,写这篇文章主要是给和我一样的新手朋友提供两种导出EXCEL的方法并探讨一下导出的效率问题,本文中的代码直接就可用,其中部分代码参考其他的代码并做了修改,抛砖引玉,希望大家一起探讨,如有不对的地方还请大家多多包涵并指出来,我也是个新手,出错也是难免的。 首先先总结下自己知道的导出EXCEL表格的方法,大致有以下几种,有疏漏的请大家补充。 1.数据逐条逐条的写入EXCEL 2.通过OLEDB把EXCEL做为数据源来写 3.通过RANGE范围写入多行多列内存数据到EXCEL 4.利用系统剪贴板写入EXCEL 好了,我想这些方法已经足够完成我们要实现的功能了,方法不在多,在精,不是么?以上4中方法都可以实现导出EXCEL,方法1为最基础的方法,意思就是效率可能不是太高,当遇到数据量过大时所要付出的时间也是巨大的,后面3种方法都是第一种的衍生,在第一种方法效率低下的基础上改进的,这里主要就是一个效率问题了,当然如果你数据量都很小,我想4种方法就代码量和复杂程度来说第1种基本方法就可以了,或当你的硬件非常牛逼了,那再差的方法也可以高效的完成也没有探讨的实际意义了,呵呵说远了,本文主要是在不考虑硬件或同等硬件条件下单从软件角度出发探讨较好的解决方案。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值