源码网商城,靠谱的源码在线交易网站 我的订单 购物车 帮助

源码网商城

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

  • 时间:2020-05-12 12:42 编辑: 来源: 阅读:
  • 扫一扫,手机访问
摘要:ASP.NET中生成Excel遇到的问题及改进方法
先看一下方法(其中略去了一些判断和扩展): [b]生成Excel老代码 [/b]
[u]复制代码[/u] 代码如下:
/// <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的内容,改进后的代码如下: [b]改进后生成Excel的代码 [/b]
[u]复制代码[/u] 代码如下:
/// <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。
  • 全部评论(0)
联系客服
客服电话:
400-000-3129
微信版

扫一扫进微信版
返回顶部