前些天写了个导出Excel的公共类,因为项目中也需要上传Excel,没有利用Office组件,(PS:Office在上传文件时候,Excel进程无法关闭,会导致上传和打开失败)有的说利用Kill把进程给关闭,我总觉得这样的方法不好,没有任何原因,关闭一个进程,想想都有点悬。

这个例子是采用NPOI组件,下载地址:http://npoi.codeplex.com/,先不说组件的事,说下实现思路。

把类实体转换表格时候,最头疼莫过于表头和字段的设计,刚开始有几个同事利用解析Html,有几个导出功能就写几个导出函数,最后函数多了就直接构建一个类,这样功能也能在实现,但不是个很好的选择。

在类转为表的时候我们可以考虑利用反射功能把类的属性和值都取出来,这个的话,我们只需要构建一个函数,利用泛型把相应的类型传入就行。

下面我们可以定义一个函数的签名:

public static void ExportExcel<T>(List<T> tList)
{
}

好了,这样的话T就是我们要处理的类型,利用反射取出字段和值

XSSFWorkbook xssfWorkbook = new XSSFWorkbook();//定义一个工作薄
XSSFSheet xssfSheet = (XSSFSheet)xssfWorkbook.CreateSheet("sheet1");//定义一个表单
NPOI.SS.UserModel.IRow titleRow = xssfSheet.CreateRow();//创建一个行
PropertyInfo[] propertyInfos = typeof(T).GetProperties();//取出字段数组
;//定义索引
  foreach (var propertyInfo in propertyInfos)//对数组进行遍历
      {
            titleRow.CreateCell(index++).SetCellValue(propertyInfo. Name);//赋值
     }

我们这样已经把字段取出来了,但是现现在的字段就类的字段,有的时候我们需要定义我们想要的表头,大多数的表头还是中文,还有的我也不要全部反射出来?这个怎么办?(解决方法在后面,千万不要放弃阅读)

好了,表头基本上已经搞定,还有一个悬念,后面我会说。下面是取出数据

], null);//取出第0 行数据
if (obj != null)
{
Row.CreateCell(index++).SetCellValue(obj.ToString());//设定到相应的列}
else
{
            index++;
}

一个字段的值取出来了,下面加快工作,代码如下:

 ;j<tList.Count();j++)//传入的List遍历
            {
                NPOI.SS.UserModel.IRow Row = xssfSheet.CreateRow(j+);//创建列

               index = ;//重新设定索引

                foreach (var propertyInfo in propertyInfos)
                {
                    Object[] exportExcelAttributes = propertyInfo.GetCustomAttributes(true);
                    )
                    {
                        foreach (var exportExcelAttribute in exportExcelAttributes)
                        {
                            var attribute = exportExcelAttribute as ExportExcelAttribute;
                            if (attribute != null)
                            {
                             object obj = propertyInfo.GetValue(tList[j], null);//取出数据
                                if (obj != null)
                                {
                                    Row.CreateCell(index++).SetCellValue(obj.ToString());//设定到相应的列
                                }
                                else
                                {
                                    index++;
                                }
                            }
                        }
                    }
                }
            }

到目前已经把一个类实体转为一个Excel了,下面开始解决刚才的那个的疑问。有的表头是中文,有的我想设定的表头和字段不一样怎么办?还有字段我不想全部导出怎么办?

这个时候我们可以考虑使用Attribute (特性),我们可以自定一个特性。

所有特性类型都直接或间接地从 Attribute 类派生。 特性可应用于任何目标元素;多个特性可应用于同一目标元素;并且特性可由从目标元素派生的元素继承。 使用AttributeTargets 类可以指定特性所应用到的目标元素。 -----MSDN

这个话不读也没事,我们只需要第一句,我们要派生一个Attribute类,代码如下:

[AttributeUsageAttribute(AttributeTargets.All, Inherited = true, AllowMultiple = false)]
   public class ExportExcelAttribute:Attribute
    {
       private readonly string titleName;

       public ExportExcelAttribute(string name)
       {
           titleName = name;
       }

       public string TitleName
       {
           get { return titleName; }
       }
    }

代码简单,不多说。这样的话类实体也要修改下,先看下类实体的变化:

public class DriverInfo : BaseEntity
    {
        [ExportExcel("姓名")]
        public string name { get; set; }
        public string DriverInfoCode { get; set; }
        [ExportExcel("备注")]
        public string note { get; set; }
        [ExportExcel("身份证号")]
        public string IDCardNum { get; set; }

        public string DriverNum { get; set; }//驾驶证号
        [ExportExcel("出生日期")]
        public DateTime? BirthDay { get; set; }
        [ExportExcel("性别")]
        public string Sex { get; set; }
        public string Location { get; set; }//归属地
    }

看了就该有了三分明白了,我们要在需要导出的字段上加上    [ExportExcel("出生日期")],用来和其它字段区别,有这个特性的我们导出,按着后面给出的名称来设定。下面工作又要回到反射上面:

生成表头:

    foreach (var propertyInfo in propertyInfos)//对数组进行遍历
            {
                Object[] exportExcelAttributes = propertyInfo.GetCustomAttributes(true);
                )
                {
                    foreach (var exportExcelAttribute in exportExcelAttributes)
                    {
                        var attribute = exportExcelAttribute as ExportExcelAttribute;
                        if (attribute != null)
                        {
                            titleRow.CreateCell(index++).SetCellValue(attribute.TitleName);
                        }
                    }
                }
            }
    

取值也是同样,代码如下:

  ;j<tList.Count();j++)//传入的List遍历
            {
                NPOI.SS.UserModel.IRow Row = xssfSheet.CreateRow(j+);//创建列

               index = ;//重新设定索引

                foreach (var propertyInfo in propertyInfos)
                {
                    Object[] exportExcelAttributes = propertyInfo.GetCustomAttributes(true);
                    )
                    {
                        foreach (var exportExcelAttribute in exportExcelAttributes)
                        {
                            var attribute = exportExcelAttribute as ExportExcelAttribute;
                            if (attribute != null)
                            {
                             object obj = propertyInfo.GetValue(tList[j], null);//取出数据
                                if (obj != null)
                                {
                                    Row.CreateCell(index++).SetCellValue(obj.ToString());//设定到相应的列
                                }
                                else
                                {
                                    index++;
                                }
                            }
                        }
                    }
                }
            }
   这样的话每次要导出数据的时候,我们只需要修改类实体就行了,而不用去做不同的处理。到此,我们的工作完成一半,说好的导出呢?

下面可以来说下导出:说下思路,Excel构建完成后,先把它保存服务器上面的一个相对路径,然后通过Response.Flush();实现下载。

重构下函数签名:

public static void ExportExcel<T>(List<T> tList, HttpContext httpContext)
{
}

代码:

    string DownPath=httpContext.Server.MapPath("~/DownloadPath");//服务器的相对径
            MemoryStream memoryStream = new MemoryStream();
        xssfWorkbook.Write(memoryStream);
            string fileName = DownPath + "\\" + DateTime.Now.ToString("yyyyMMddHHssmm") + "_"+typeof(T).Name + ".xlsx";  //定义文件名称和路径
        File.WriteAllBytes(fileName, memoryStream.ToArray());//把内存内容写入文件

下面工作就是把文件读取,实现下载:

      string fName = new FileInfo(fileName).Name;
            FileStream fs = new FileStream(fileName, FileMode.Open);
            byte[] bytes = new byte[(int)fs.Length];
            fs.Read(bytes, , bytes.Length);
            fs.Close();
            httpContext.Response.Charset = "UTF-8";
            httpContext.Response.ContentEncoding = System.Text.Encoding.GetEncoding("UTF-8");
            httpContext.Response.ContentType = "application/octet-stream";

            httpContext.Response.AddHeader("Content-Disposition", "attachment; filename=" + fName);
            httpContext.Response.BinaryWrite(bytes);
            httpContext.Response.Flush();
            httpContext.Response.End();//下载

            if (File.Exists(fileName))//完成后删除
            {
                File.Delete(fileName);
            }
      </pre><pre code_snippet_id="371974" snippet_file_name="blog_20140531_9_5695766" name="code" class="csharp">

导出结果:

到此,这个功能算是写完了,我制作个Demo下载:

http://download.csdn.net/detail/fw199006/7430623

有兴趣加群    IT项目 324110381




												

第十篇 一个利用反射实现的Excel导出的更多相关文章

  1. Java开发笔记(七十九)利用反射技术操作私有属性

    早在介绍多态的时候,曾经提到公鸡实例的性别属性可能被篡改为雌性,不过面向对象的三大特性包含了封装.继承和多态,只要把性别属性设置为private私有级别,也不提供setSex这样的性别修改方法,那么性 ...

  2. 一个很好的通用 excel 导出工具类

    此类用主要 jxl +注解+流 实现扩展性很强,jxl性能会比poi好一点,值得我们学习. package oa.common.utils; import java.io.OutputStream; ...

  3. 一个把数据转化成Excel导出的程序 python Django

    把从数据库查询出来数据导出 源码下载!!!!! 效果图 登入界面 主页面 查询到数据 导出 打开得到文件 项目地址,源码下载

  4. 利用反射跟自定义注解拼接实体对象的查询SQL

    前言 项目中虽然有ORM映射框架来帮我们拼写SQL,简化开发过程,降低开发难度.但难免会出现需要自己拼写SQL的情况,这里分享一个利用反射跟自定义注解拼接实体对象的查询SQL的方法. 代码 自定义注解 ...

  5. Python基础篇【第3篇】: Python异常处理、反射、动态导入、利用反射的web框架

    异常处理 什么是异常? 异常即是一个事件,该事件会在程序执行过程中发生,影响了程序的正常执行. 一般情况下,在Python无法正常处理程序时就会发生一个异常.异常是Python对象,表示一个错误.当P ...

  6. C# 调用一个按钮的Click事件(利用反射)

    最基本的调用方法 (1)button1.PerformClick();(2)button1_Click(null,null);(3)button_Click(null,new EventArgs()) ...

  7. Java开发笔记(八十)利用反射技术操作私有方法

    前面介绍了如何利用反射技术读写私有属性,不单是私有属性,就连私有方法也能通过反射技术来调用.为了演示反射的逆天功能,首先给Chicken鸡类增加下列几个私有方法,简单起见弄来了set***/get** ...

  8. 利用反射--调用一个按钮的Click事件

    最基本的调用方法 (1)button1.PerformClick();(2)button1_Click(null,null);(3)button_Click(null,new EventArgs()) ...

  9. C#利用反射,遍历获得一个类的所有属性名,以及该类的实例的所有属性的值

    转自goldeneyezhang原文 C#利用反射,遍历获得一个类的所有属性名,以及该类的实例的所有属性的值 C#利用反射,遍历获得一个类的所有属性名,以及该类的实例的所有属性的值总结: 对应某个类的 ...

随机推荐

  1. bzoj3991 [Sdoi2015]寻宝游戏 set动态维护虚树+树链求并

    题目大意:支持多次操作,增加或删除一个关键点 动态维护虚树边权和*2 分析:可以用树链求并的方法,最后减去虚树的根到1距离 注意到树链求并是所有点到根距离-所有dfn序相邻两点的LCA到根距离 找df ...

  2. php模式设计之 策略模式

    策略模式: 策略模式设计帮助构建的对象不必自身包含逻辑,而是能够根据需要利用其他对象中的算法. 使用场景: 例如有一个CD类,我们类存储了CD的信息. 原先的时候,我们在CD类中直接调用getCD方法 ...

  3. easyui datagrid分页参数获取

    $("#btnDataExport").click(function () { $.messager.show({ title: '导出提示', msg: '每次导出一页商户数据, ...

  4. SQL server存储过程:数据的插入和更新

    存储过程的功能非常强大,在某种程度上甚至可以替代业务逻辑层, 接下来就一个小例子来说明,用存储过程插入或更新语句. 1.数据库表结构 所用数据库为Sql Server2008. 2.创建存储过程 (1 ...

  5. jQuery event,冒泡,默认事件用法

    jQuery event,冒泡,默认事件用法 <%@ page language="java" import="java.util.*" pageEnco ...

  6. CaltrainTimes从设计到发布(基于Flex的手机应用)

    资源下载地址 Caltrain Times 的 iTunes下载地址 Caltrain Times的Android Market 下载地址 Caltrain Times的BlackBerry App ...

  7. [MyBatis]mapperLocations属性通配符的使用

    http://blog.csdn.net/szwangdf/article/details/23432783 http://ljhzzyx.blog.163.com/blog/static/38380 ...

  8. LoadRunner相关架构图

    LoadRunner概览图: Lr架构图:

  9. 从php到浏览器的缓存机制,不得不看!

    所有的php程序员都知道在php脚本里面执行 echo "1";访客的浏览器里面就会显示"1". 但是我们执行下面的代码的时候,并不是显示"1&quo ...

  10. 简单的java Hadoop MapReduce程序(计算平均成绩)从打包到提交及运行

    [TOC] 简单的java Hadoop MapReduce程序(计算平均成绩)从打包到提交及运行 程序源码 import java.io.IOException; import java.util. ...