欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

一个通用的List集合导出excel的通用方法

程序员文章站 2022-03-25 17:18:38
前几天要做一个数据导出Excel 我就打算写一个通用的。 这样一来用的时候也方便,数据主要是通过Orm取的List。这样写一个通用的刚好。 public static void ListToExcel(List ts, string[] RowName, string[] List ......

原文地址:https://www.cnblogs.com/silentCM/p/12711219.html

前几天要做一个数据导出excel 

我就打算写一个通用的。

这样一来用的时候也方便,数据主要是通过orm取的list。这样写一个通用的刚好。

        public static void listtoexcel(list<dynamic> ts, string[] rowname, string[] listcorrespondrow, bool isrowname = false)
        {

            //创建工作簿对象
            iworkbook workbook = new hssfworkbook();
            //创建工作表
            isheet sheet = workbook.createsheet("onesheet");
            irow row0 = sheet.createrow(0);
            for (int i = 0; i < rowname.length; i++)
            {
                row0.createcell(i).setcellvalue(rowname[i]);
            }
            for (int r = 1; r <= ts.count; r++)
            {
                //创建行row
                irow row = sheet.createrow(r);
                dynamic tsc = ts[r-1];string sjson = jsonconvert.serializeobject(tsc);
                dynamic sobj = jsonconvert.deserializeobject<dynamic>(sjson);
                var sobjlen = sobj.gettype().getproperties();
                for (int j = 0; j < listcorrespondrow.length; j++)
                {
                    //通过【】来取值  但必须要通过json转成的对象才可以这样取
                    row.createcell(j).setcellvalue(convert.tostring(sobj[listcorrespondrow[j]]));
                }
                //foreach (system.reflection.propertyinfo p in tsc.gettype().getproperties())
                //{
                //    //row.createcell().setcellvalue(p.getvalue(tsc));
                //}

                //for (int j = 0; j < sobjlen.length; j++)
                //{
                    //利用反射将对象里面的值添加到excel里面 添加的顺序是按照对象里面字段的顺序 注意和列名保持一致
                //    row.createcell(j).setcellvalue(sobjlen[j].getvalue(tsc));
                //}

            }


            //创建流对象并设置存储excel文件的路径
            using (filestream url = file.openwrite(@"c:/users/13002/source/repos/练习/练习/worddot/test3.xls"))
            {

                //导出excel文件
                workbook.write(url);
                //response.write("<script>alert('写入成功!')</script>");
            };

            ////workbook.write();
            ////创建文件流
            //memorystream bookstream = new memorystream();
            ////文件写入流(向流中写入字节序列)
            //workbook.write(bookstream);
            ////输出之前调用seek(偏移量,游标位置) 把0位置指定为开始位置
            //bookstream.seek(0, seekorigin.begin);
            //return bookstream;
        }

在写这个的时候就遇到了一些问题。

刚开始是打算用反射进去获取,因为刚开始我自己试了一下(我手动创建了一个list集合里面的对象也是自己手动输入的)

这个时候用

foreach (system.reflection.propertyinfo p in tsc.gettype().getproperties())
{
    row.createcell().setcellvalue(p.getvalue(tsc));
}

这串代码来往excel里面插入是没有问题的。

但是后来发现我自己创建的list和数据库查询之后返回的list不一样。

我数据库框架用的dapper,接受集合的时候用的是list<dynamic>

这时候就用反射获取不到有多少个属性了,也就取不到值了。

后来我想既然这样我就把他转成json在把他转成dynamic。

后来试了一下,果然可以获取的到属性长度的数组。

但是不能用foreach,因为这样会出错,给excel每一列赋值的时候需要传索引号。

一个通用的List集合导出excel的通用方法

   我也就是我单独把他拉出来的原因。

一个通用的List集合导出excel的通用方法一个通用的List集合导出excel的通用方法

   但是这样用循环依次获取属性的值会出问题,会报错。

 

 

一个通用的List集合导出excel的通用方法

   然后我只得用这个方法了。用这样的话,还需要自己定义一个数组把当前对象有字段的名称告诉这个方法,所以略显麻烦,所以之前一直在搞不要输入的按照顺序直接赋值的。但是没弄出来:)

之所以能用这个方法是因为把对象转成json在把json转成对象后这个对象是jobject 就是newtonsoft.json里面的一个东西。他支持用【】来获取数据

dynamic是不支持【】获取属性的值的。

还有就是setcellvalue不加convert.tostring有时候会报错,报具有二义性,我f12看了一下源码,

一个通用的List集合导出excel的通用方法

应该是这两个有点小差异,所以转换一下就好了

一个通用的List集合导出excel的通用方法

   与用法就是这个样子的