ASP.NET中生成Excel遇到的问题及改进方法
2023-06-13 09:14:27 时间
先看一下方法(其中略去了一些判断和扩展):
生成Excel老代码
复制代码代码如下:
生成Excel老代码
///<summary>
///将一组对象导出成EXCEL
///</summary>
///<typeparamname="T">要导出对象的类型</typeparam>
///<paramname="objList">一组对象</param>
///<paramname="FileName">导出后的文件名</param>
///<paramname="columnInfo">列名信息</param>
publicstaticvoidExExcel<T>(List<T>objList,stringFileName,Dictionary<string,string>columnInfo)
{
if(columnInfo.Count==0){return;}
if(objList.Count==0){return;}
//生成EXCEL的HTML
stringexcelStr="";
TypemyType=objList[0].GetType();
//根据反射从传递进来的属性名信息得到要显示的属性
List<PropertyInfo>myPro=newList<PropertyInfo>();
foreach(stringcNameincolumnInfo.Keys)
{
PropertyInfop=myType.GetProperty(cName);
if(p!=null)
{
myPro.Add(p);
excelStr+=columnInfo[cName]+"\t";
}
}
//如果没有找到可用的属性则结束
if(myPro.Count==0){return;}
excelStr+="\n";
foreach(TobjinobjList)
{
foreach(PropertyInfopinmyPro)
{
excelStr+=p.GetValue(obj,null)+"\t";
}
excelStr+="\n";
}
//输出EXCEL
HttpResponsers=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>
///<typeparamname="T">要导出对象的类型</typeparam>
///<paramname="objList">一组对象</param>
///<paramname="FileName">导出后的文件名</param>
///<paramname="columnInfo">列名信息</param>
publicstaticvoidExExcel<T>(List<T>objList,stringFileName,Dictionary<string,string>columnInfo)
{
if(columnInfo.Count==0){return;}
if(objList.Count==0){return;}
//生成EXCEL的HTML
StringBuilderexcelStr=newStringBuilder(objList.Count*columnInfo.Count);
TypemyType=objList[0].GetType();
//根据反射从传递进来的属性名信息得到要显示的属性
List<PropertyInfo>myPro=newList<PropertyInfo>();
foreach(stringcNameincolumnInfo.Keys)
{
PropertyInfop=myType.GetProperty(cName);
if(p!=null)
{
myPro.Add(p);
excelStr.Append(columnInfo[cName]).Append("\t");
}
}
//如果没有找到可用的属性则结束
if(myPro.Count==0){return;}
excelStr.Append("\n");
foreach(TobjinobjList)
{
foreach(PropertyInfopinmyPro)
{
excelStr.Append(p.GetValue(obj,null)).Append("\t");
}
excelStr.Append("\n");
}
//输出EXCEL
HttpResponsers=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();
}
}
在实例化StringBuilderexcelStr=newStringBuilder(objList.Count*columnInfo.Count);时候预分配开始大小,这样能更好的使用StringBuilder。至此,改进完成。
另外,如果您觉得反射会影响性能,那么可以改成表达式树的方式,或者使用limit。
相关文章
- 导出的Excel名字乱码_恢复的excel文件乱码
- [接上篇]在Window10/11的Linux子系统Docker上部署VB.NET Asp.Net Core WebAPI应用
- .Net Core3.1 SignalR for WPF Asp.net
- 数据库的比较深入比较:Excel与MySQL的数据库技术优劣(excel与mysql)
- 轻松操作:oracle快速导入Excel数据(oracle导入excel数据)
- Oracle Database: The Ultimate Connection Guide for .NET Developers(net连接oracle)
- asp.net的web服务MSSQL检测ASP.NET的Web服务——利用它构建更棒的站点(mssql检测基于)
- 数据库NET开发者操作Oracle数据库的全攻略(net操作oracle)
- NET技术结合MySQL实现数据库访问极致体验(.net数据库mysql)
- asp.net操作excel的实现代码
- asp.net下用Aspose.Wordsfor.NET动态生成word文档中的数据表格的方法
- 浅析ASP.NET安全性分析(加强asp.net1.1/2.0安全性)
- .NET中基于事件的异步模式-EAP
- asp.net读取excel文件的三种方法示例
- ASP.NET使用GridView导出Excel实现方法
- asp.net中Table生成Excel表格的方法
- Asp.Net使用Npoi导入导出Excel的方法