当前位置: 移动技术网 > IT编程>开发语言>.net > ASP.NET中生成Excel遇到的问题及改进方法

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

2017年12月12日  | 移动技术网IT编程  | 我要评论

野骜射手,倪睿思近况,蜡像馆丧尸

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

如对本文有疑问,请在下面进行留言讨论,广大热心网友会与你互动!! 点击进行留言回复

相关文章:

验证码:
移动技术网