ASP.NET中生成Excel遇到的问题及改进方法

 更新时间:2011年02月19日 17:28:59   作者:  
这是在项目中使用的生成Excel的一个方法(其实是我在网上收集),用了半年时间,并未出现异常情况。近日当我再次使用该方法生成Excel的时候出现了问题
先看一下方法(其中略去了一些判断和扩展):
生成Excel老代码
复制代码 代码如下:

/// <summary>
/// 将一组对象导出成EXCEL
/// </summary>
/// <typeparam name="T">要导出对象的类型</typeparam>
/// <param name="objList">一组对象</param>
/// <param name="FileName">导出后的文件名</param>
/// <param name="columnInfo">列名信息</param>
public static void ExExcel<T>(List<T> objList, string FileName, Dictionary<string, string> columnInfo)
{

if (columnInfo.Count == 0) { return; }
if (objList.Count == 0) { return; }
//生成EXCEL的HTML
string excelStr = "";

Type myType = objList[0].GetType();
//根据反射从传递进来的属性名信息得到要显示的属性
List<PropertyInfo> myPro = new List<PropertyInfo>();
foreach (string cName in columnInfo.Keys)
{
PropertyInfo p = myType.GetProperty(cName);
if (p != null)
{
myPro.Add(p);
excelStr += columnInfo[cName] + "\t";
}
}
//如果没有找到可用的属性则结束
if (myPro.Count == 0) { return; }
excelStr += "\n";

foreach (T obj in objList)
{
foreach (PropertyInfo p in myPro)
{
excelStr += p.GetValue(obj, null) + "\t";
}
excelStr += "\n";
}

//输出EXCEL
HttpResponse rs = System.Web.HttpContext.Current.Response;
rs.ContentEncoding = System.Text.Encoding.GetEncoding("GB2312");
rs.AppendHeader("Content-Disposition", "attachment;filename=" + FileName);
rs.ContentType = "application/ms-excel";
rs.Write(excelStr);
rs.End();
}

到这个时候我想应该有朋友能看出来问题所在了。

这个方法生成Excel数据量不大的时候不会出现问题,当数据量变大之后问题就出来了。因为方法里面定义了一个string类型的变量,将需要填充到Excel的内容叠加。对于string类型的数据使用+=操作相当于使用string.Concat方法连接字符串。每当进行一次+=操作的时候就会生成一个新字符串。必然会开辟一块内存,这样的操作一多就会把内存耗尽,产生一个OutOfMemoryException。

知道了问题所在,改进起来也很容易,那就是利用StringBuilder叠加需要填充到Excel的内容,改进后的代码如下:
改进后生成Excel的代码
复制代码 代码如下:

/// <summary>
/// 将一组对象导出成EXCEL
/// </summary>
/// <typeparam name="T">要导出对象的类型</typeparam>
/// <param name="objList">一组对象</param>
/// <param name="FileName">导出后的文件名</param>
/// <param name="columnInfo">列名信息</param>
public static void ExExcel<T>(List<T> objList, string FileName, Dictionary<string, string> columnInfo)
{

if (columnInfo.Count == 0) { return; }
if (objList.Count == 0) { return; }
//生成EXCEL的HTML
StringBuilder excelStr = new StringBuilder(objList.Count * columnInfo.Count);

Type myType = objList[0].GetType();
//根据反射从传递进来的属性名信息得到要显示的属性
List<PropertyInfo> myPro = new List<PropertyInfo>();
foreach (string cName in columnInfo.Keys)
{
PropertyInfo p = myType.GetProperty(cName);
if (p != null)
{
myPro.Add(p);
excelStr.Append(columnInfo[cName]).Append("\t");
}
}
//如果没有找到可用的属性则结束
if (myPro.Count == 0) { return; }
excelStr.Append("\n");

foreach (T obj in objList)
{
foreach (PropertyInfo p in myPro)
{
excelStr.Append(p.GetValue(obj, null)).Append("\t");
}
excelStr.Append("\n");
}

//输出EXCEL
HttpResponse rs = System.Web.HttpContext.Current.Response;
rs.ContentEncoding = System.Text.Encoding.GetEncoding("GB2312");
rs.AppendHeader("Content-Disposition", "attachment;filename=" + FileName);
rs.ContentType = "application/ms-excel";
rs.Write(excelStr);
rs.End();
}
}

在实例化StringBuilder excelStr = new StringBuilder(objList.Count * columnInfo.Count);时候预分配开始大小,这样能更好的使用StringBuilder。至此,改进完成。
另外,如果您觉得反射会影响性能,那么可以改成表达式树的方式,或者使用limit。

相关文章

  • .net压缩功能实现方法

    .net压缩功能实现方法

    这篇文章主要介绍了.net压缩功能实现方法,需要的朋友可以参考下
    2014-02-02
  • .Net实现延迟队列

    .Net实现延迟队列

    这篇文章介绍了.Net实现延迟队列的方法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-07-07
  • iis 服务器应用程序不可用的解决方法

    iis 服务器应用程序不可用的解决方法

    访问页面时提示 服务器应用程序不可用,大家可以按照下面的方法重新注册下,应该能好点
    2008-11-11
  • ASP.NET MVC使用typeahead.js实现输入智能提示功能

    ASP.NET MVC使用typeahead.js实现输入智能提示功能

    这篇文章介绍了ASP.NET MVC使用typeahead.js实现输入智能提示功能的方法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-09-09
  • asp.net core集成ElasticSearch实现全文检索功能

    asp.net core集成ElasticSearch实现全文检索功能

    索引是Elasticsearch中用于存储文档的容器,你可以使用Elasticsearch的REST API、官方客户端库(如NEST)或Kibana等工具来创建和管理索引,本文给大家介绍asp.net core集成ElasticSearch实现全文检索功能,感兴趣的朋友一起看看吧
    2024-08-08
  • 在ASP.NET Core 中发送邮件的实现方法(必看篇)

    在ASP.NET Core 中发送邮件的实现方法(必看篇)

    下面小编就为大家带来一篇在ASP.NET Core 中发送邮件的实现方法(必看篇)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-05-05
  • ASP.NET中水晶报表的使用方法详解

    ASP.NET中水晶报表的使用方法详解

    这篇文章介绍了ASP.NET中水晶报表的使用方法,有需要的朋友可以参考一下
    2013-11-11
  • .NET使用Collections.Pooled提升性能优化的方法

    .NET使用Collections.Pooled提升性能优化的方法

    这篇文章主要介绍了.NET使用Collections.Pooled性能优化的方法,今天要给大家分享类库Collections.Pooled,它是通过池化内存来达到降低内存占用和GC的目的,另外也会带大家看看源码,为什么它会带来这些性能提升,一起通过本文学习下吧
    2022-05-05
  • ASP.NET MVC实现依赖注入的完整过程

    ASP.NET MVC实现依赖注入的完整过程

    这篇文章主要给大家介绍了关于ASP.NET MVC实现依赖注入的完整过程,文中通过示例代码介绍的非常详细,对大家学习或者使用ASP.NET MVC具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-09-09
  • aspxgridview CustomButtonCallback 不支持弹出消息提示解决方法

    aspxgridview CustomButtonCallback 不支持弹出消息提示解决方法

    aspxgridveiw是devexpress的一个grid控件,使用起来还不错,不能再 CustomButtonCallback 事件中使用response.write,具体的解决方法如下,感兴趣的朋友可以参考下哈
    2013-06-06

最新评论