前几天要在后台查询数据库内容(用entity framework),将查询出来的信息(List或DataTable形式)转成EXCEL供用户下载。经过谷歌、百度搜索,终于搜出了一些代码。似乎可用了,结果问题一大堆。

/// <summary>
    /// 下载EXCEL
    /// </summary>
    /// <param name="dt">数据</param>
    /// <param name="fileName">下载的EXCEL文件名</param>
    public static void OutputToExcelGB2312(object dt, string fileName)
    {
        if (dt != null)
        {
            // 获取当前会话,并设置当前会话中Response的相关属性,让输出内容输出到文件中。
            System.Web.HttpContext curContext = System.Web.HttpContext.Current;
            curContext.Response.Charset = "GB2312";
            curContext.Response.ContentEncoding = System.Text.Encoding.GetEncoding("GB2312");
            curContext.Response.AddHeader("content-disposition", "attachment;filename="
             + System.Web.HttpUtility.UrlEncode(fileName, System.Text.Encoding.UTF8) + ".xls");
            curContext.Response.ContentType = "application/ms-excel";

            //将要导出的数据绑定到DataGrid控件上
            System.Web.UI.WebControls.DataGrid dgExport = new System.Web.UI.WebControls.DataGrid();
            dgExport.DataSource = dt;
            dgExport.AllowPaging = false;
            dgExport.DataBind();
            //创建输出流用来保存输出内容
            System.IO.StringWriter strWriter = new System.IO.StringWriter();
            System.Web.UI.HtmlTextWriter htmlWriter = new System.Web.UI.HtmlTextWriter(strWriter);

            //将DataGrid控件的内容输出到流中。
            dgExport.RenderControl(htmlWriter);

            curContext.Response.Write(strWriter.ToString());
            curContext.Response.End();
        }
    }

参数为DataTable形式的数据和str文件名。这个函数主要意思是利用dataGrid这类控件生成行列信息,最终转至EXCEL类型。

但这个函数通用性不好,因为我用entity framework或者LINQ查出来的数据大多是List或IEnumable泛型形式的,不便转换成DataTable形式(其实可以转,就是定义个DataTable dt;添加列名,最后定义行并填充数据,最后提交dt即可),是在麻烦,特别是列数据一多,代码杂糅,很是不爽。

我想, dataGrid这类控件不是有根据数据源自动生成列及填充数据项的功能吗?!那我就不用手动做DataTable定义列并赋值行了,让dataGrid自动生成吧!

MSDN上关于DataGrid或GridView这类控件的DataSource属性是这么说明的:

“使用 DataSource 属性指定要绑定到数据列表控件的值的源。 数据源必须是实现 System.Collections .IEnumerable 接口(例如 System.Data .DataViewSystem.Collections .ArrayListSystem.Collections .Hashtable)或 IListSource 接口的对象,才能绑定到从 BaseDataList 类派生的控件。 在设置 DataSource

必须手动编写代码才能执行数据绑定。”

也就是说,只要数据源(object形式)支持IEnumerable迭代,那控件就可以根据数据源自动生成列、填充行数据等。既然如此,我用entity framework或者LINQ查出的List或IEnumable泛型形式的数据当然可以生成啦!

下面把上述方法稍改造下,使之可以支持任意IEnumable接口的数据源:

  /// <summary>
     /// 下载EXCEL
     /// </summary>
     /// <param name="dt">数据</param>
     /// <param name="fileName">下载的EXCEL文件名</param>
     public static void OutputToExcelGB2312(object dt, string fileName)
     {
         if (dt != null)
         {
             // 获取当前会话,并设置当前会话中Response的相关属性,让输出内容输出到文件中。
             System.Web.HttpContext curContext = System.Web.HttpContext.Current;
             curContext.Response.Charset = "GB2312";
             curContext.Response.ContentEncoding = System.Text.Encoding.GetEncoding("GB2312");
             curContext.Response.AddHeader("content-disposition", "attachment;filename="
              + System.Web.HttpUtility.UrlEncode(fileName, System.Text.Encoding.UTF8) + ".xls");
             curContext.Response.ContentType = "application/ms-excel";

             //将要导出的数据绑定到DataGrid控件上
             System.Web.UI.WebControls.DataGrid dgExport = new System.Web.UI.WebControls.DataGrid();
             dgExport.DataSource = dt;
             dgExport.AllowPaging = false;
             dgExport.DataBind();
             //创建输出流用来保存输出内容
             System.IO.StringWriter strWriter = new System.IO.StringWriter();
             System.Web.UI.HtmlTextWriter htmlWriter = new System.Web.UI.HtmlTextWriter(strWriter);

             //将DataGrid控件的内容输出到流中。
             dgExport.RenderControl(htmlWriter);

             curContext.Response.Write(strWriter.ToString());
             curContext.Response.End();
         }
     }

这样一些,清爽多了。

写个例子吧,各位如果有幸看到这些代码,算是抛砖引玉,给予启发了。

1.新建ASP.NET项目。加个窗体,在后台写代码,为了方便就在Page_Load()方法里写代码了。

 using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Web;
 using System.Web.UI;
 using System.Web.UI.WebControls;
 public class Info
 {
     public int a { get; set; }
     public Nullable<int> b { get; set; }

 }
 public partial class ExcelTest : System.Web.UI.Page
 {
     protected void Page_Load(object sender, EventArgs e)
     {
         Info i = , b =  };
         Info j = , b =  };
         List<Info> l = new List<Info>() { i, j };
         OutputToExcelGB2312(l, "test");

         //DataGrid1.DataSource = l;
         //DataGrid1.DataBind();
     }
 }

浏览运行,跳出下载对话框,下载EXCEL。以为大功告成,结果:

见鬼!!!!!只有只有列a,列b哪去了?!!!

多次调试,确认问题在OutputToExcelGB2312()方法里:

         System.Web.UI.WebControls.DataGrid dgExport = new System.Web.UI.WebControls.DataGrid();
             dgExport.DataSource = dt;
             dgExport.AllowPaging = false;
             dgExport.DataBind();  //查看属性的值,这时自动生成的列数是1,及 a,没有b.

我当时一时难理解,为什么?!突然瞥见我查数据库信息是有的信息是int?及Nullable<int>类型的。这ORM关系(entity framework,linq)很常见,因为数据库里有些字段是可以为空的,这些字段ORM转换后就成为了Nullable<T>类型。

在这个例子里就是数据项类型是:

public Nullable<int> b { get; set; }

难道是dataGrid不识别Nullable<int>?!果断换个声明public int b { get; set; }就这样,问题解决了。。。。

不过,我依然只知大概,不知为什么(内部绑定原理比较复杂).应该有反射、动态绑定、IEnumable、数据项生成等知识吧。大家可以百度谷歌下,例如以下资料:http://kb.cnblogs.com/kb/70077/

个人理解dataGrid是.NET1.0就有的东西,那时Nullable类型还没有出现,所以可能不认识(dadaGrid向后兼容有点问题吧)基于此,我想到还有另外一种方法,在OutputToExcelGB2312()中将dataGrid改成GridView,这样Gridview就认识Nullable类型,当然就可以实现了(但是Gridview会对bool等类型转换,成为checkBox类型。)有点自作主张了,最后转化为EXCEL时里面就会存在checkBox,原数据为true则打钩,false则不打勾。

.NET细节隐藏的太好,新手入门很简单,拖拖控件即可。但开源的东西较少,有时用的多了真是迷糊了,不知内部到底怎么实现的,结果错漏摆出。哎!!看来要好好学习了,适当研究研究内部源码。包括C#,CLR,IL等东西了,只有这样才能抓住本质,深刻理会;同时不易犯错,程序才能高效稳定,程序员也能省省心,提高效率!
												





											

ASP.NET网页生成EXCEL并下载(利用DataGrid或GridView等)的更多相关文章

  1. Asp.net MVC 简单实现生成Excel并下载

    由于项目上的需求,需要导出指定条件的Excel文件.经过一翻折腾终于实现了. 现在把代码贴出来分享 (直接把我们项目里面的一部份辅助类的代码分享一下) 我们项目使用的是Asp.Net MVC4.0模式 ...

  2. 使用node.js生成excel报表下载(excel-export express篇)

    引言:日常工作中已经有许多应用功能块使用了nodejs作为web服务器,而生成报表下载也是我们在传统应用. java中提供了2套类库实现(jxl 和POI),.NET 作为微软的亲儿子更加不用说,各种 ...

  3. asp.net+nopi生成Excel遇到设置单元格值null问题

    Npoi 生成excel报表功能很不错,功能也不用给大家介绍了.首先看遇到的问题吧! FileStream file = new FileStream(Server.MapPath("Tem ...

  4. java动态生成excel打包下载

    @SuppressWarnings("unchecked") public String batchExport() throws DBException{ @SuppressWa ...

  5. Asp.net MVC 生成zip并下载

    前面有生成Excel或Word的示例,所以就不再重新写了. 这里只提供将指定文件以ZIP的方式下载. 创建一个 Zip工具类 public class ZIPCompressUtil { public ...

  6. phpexcel生成excel并下载

    Loader::import('PHPExcel.Classes.PHPExcel'); // tp5中只需将phpexcel文件放入extend文件夹中,即可采用该方法引入,需要先 use thin ...

  7. asp.net 生成 excel导出保存时, 解决迅雷下载aspx页面问题

    网络上搜索,一大堆废话,以下为简单的导出生成Excel代码: string excelFile = Server.MapPath("~/SB/UpFile/20151104111008/Bo ...

  8. JAVA利用JXL导出/生成 EXCEL

    /** * 导出导出采暖市场部收入.成本.利润明细表 * @author JIA-G-Y */ public String exporExcel(String str) { String str=Se ...

  9. .net生成Excel,并下载

    生成Excel的方式有很多种,这里记录两个最简单的: 1.将数据保存为html,然后输出到客户端,保存为Excel文件: 2.通过\t\n生成字符串,然后输出到客户端,保存为Excel. 以上两者的原 ...

随机推荐

  1. 转:工具类之SpannableStringUtils(相信你会爱上它)

    这个工具类真是构思了良久才设计出来,采用了建造者模式,然后你们就可以用链式调用了,talk is cheap, let me show the demo. demo code 有没有心动一下哈,下面就 ...

  2. HTML 学习笔记 JavaScript(数组)

    1.数组的创建 var arrayObj = new Array(); //创建一个数组var arrayObj = new Array([size]); //创建一个数组并指定长度,注意不是上限,是 ...

  3. 数据库连接池:Druid

    转自: http://www.cnblogs.com/windlaughing/p/3287501.html Java连接数据库方法概述 java.sql提供了一些接口和类,用于支持数据库增删改查等相 ...

  4. discuz论坛与其它网站登录注册整合

    discuz论坛与其它网站登录注册整合 本文以discuz 7.0.0 php版本的论坛与 .net 2.0的网站注册登录整合为类.没有采用uc_center或第三方插件.以另类的方式实现.此方法实现 ...

  5. ZOJ 2770火烧连营——差分约束

    偶尔做了一下差分约束. 题目大意:给出n个军营,每个军营最多有ci个士兵,且[ai,bi]之间至少有ki个士兵,问最少有多少士兵. ---------------------------------- ...

  6. Linux提示no crontab for root的解决办法

    Linux提示no crontab for root的解决办法 安装crontab:yum install crontabs 说明:/sbin/service crond start //启动服务/s ...

  7. ExtJS笔记5 Components

    参考 :http://blog.csdn.net/zhangxin09/article/details/6914882 An Ext JS application's UI is made up of ...

  8. 《30天自制操作系统》02_day_学习笔记

    helloos3: helloos.nas的解释在P29中 接下来课本讲了一些汇编语言的知识,便于理解这个汇编文件helloos4: 讲解在P41 helloos.nas后半部分去掉就成了ipl.as ...

  9. java位运算符常见用法

    1. 判断int型变量a是奇数还是偶数 a&1 = 0 偶数 a&1 = 1 奇数 2. 求平均值,比如有两个int类型变量x.y,首先要求x+y的和,再除以2,但是有可能x+y的结果 ...

  10. Matplotlib之无GUI时的解决办法

    需添加: import matplotlib as mpl mpl.use('Agg') 而且必须添加在import matplotlib.pyplot之前,否则无效