服务器之家

服务器之家 > 正文

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

时间:2019-09-12 11:25     来源/作者:asp.net之家

先看一下方法(其中略去了一些判断和扩展): 
生成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。

标签:

相关文章

热门资讯

玄元剑仙肉身有什么用 玄元剑仙肉身境界等级划分
玄元剑仙肉身有什么用 玄元剑仙肉身境界等级划分 2019-06-21
华为nova5pro和p30pro哪个好 华为nova5pro和华为p30pro对比详情
华为nova5pro和p30pro哪个好 华为nova5pro和华为p30pro对比详情 2019-06-22
抖音撒撒累累是什么歌 撒撒累累张艺兴歌曲名字
抖音撒撒累累是什么歌 撒撒累累张艺兴歌曲名字 2019-06-05
男生常说24816是什么意思?女生说13579是什么意思?
男生常说24816是什么意思?女生说13579是什么意思? 2019-07-24
超A是什么意思 你好a表达的是什么
超A是什么意思 你好a表达的是什么 2019-06-06
返回顶部