这是在项目中使用的生成Excel的一个方法(其实是我在网上收集),用了半年时间,并未出现异常情况。近日当我再次使用该方法生成Excel的时候出现了问题,先看一下方法(其中略去了一些判断和扩展):
生成Excel老代码
1 /// <summary>
2 /// 将一组对象导出成EXCEL
3 /// </summary>
4 /// <typeparam name="T">要导出对象的类型</typeparam>
5 /// <param name="objList">一组对象</param>
6 /// <param name="FileName">导出后的文件名</param>
7 /// <param name="columnInfo">列名信息</param>
8 public static void ExExcel<T>(List<T> objList, string FileName, Dictionary<string, string> columnInfo)
9 {
10
11 if (columnInfo.Count == 0) { return; }
12 if (objList.Count == 0) { return; }
13 //生成EXCEL的HTML
14 string excelStr = "";
15
16 Type myType = objList[0].GetType();
17 //根据反射从传递进来的属性名信息得到要显示的属性
18 List<PropertyInfo> myPro = new List<PropertyInfo>();
19 foreach (string cName in columnInfo.Keys)
20 {
21 PropertyInfo p = myType.GetProperty(cName);
22 if (p != null)
23 {
24 myPro.Add(p);
25 excelStr += columnInfo[cName] + "\t";
26 }
27 }
28 //如果没有找到可用的属性则结束
29 if (myPro.Count == 0) { return; }
30 excelStr += "\n";
31
32 foreach (T obj in objList)
33 {
34 foreach (PropertyInfo p in myPro)
35 {
36 excelStr += p.GetValue(obj, null) + "\t";
37 }
38 excelStr += "\n";
39 }
40
41 //输出EXCEL
42 HttpResponse rs = System.Web.HttpContext.Current.Response;
43 rs.ContentEncoding = System.Text.Encoding.GetEncoding("GB2312");
44 rs.AppendHeader("Content-Disposition", "attachment;filename=" + FileName);
45 rs.ContentType = "application/ms-excel";
46 rs.Write(excelStr);
47 rs.End();
48 }
到这个时候我想应该有朋友能看出来问题所在了。
这个方法生成Excel数据量不大的时候不会出现问题,当数据量变大之后问题就出来了。因为方法里面定义了一个string类型的变量,将需要填充到Excel的内容叠加。对于string类型的数据使用+=操作相当于使用string.Concat方法连接字符串。每当进行一次+=操作的时候就会生成一个新字符串。必然会开辟一块内存,这样的操作一多就会把内存耗尽,产生一个OutOfMemoryException。
知道了问题所在,改进起来也很容易,那就是利用StringBuilder叠加需要填充到Excel的内容,改进后的代码如下:
改进后生成Excel的代码
1 /// <summary>
2 /// 将一组对象导出成EXCEL
3 /// </summary>
4 /// <typeparam name="T">要导出对象的类型</typeparam>
5 /// <param name="objList">一组对象</param>
6 /// <param name="FileName">导出后的文件名</param>
7 /// <param name="columnInfo">列名信息</param>
8 public static void ExExcel<T>(List<T> objList, string FileName, Dictionary<string, string> columnInfo)
9 {
10
11 if (columnInfo.Count == 0) { return; }
12 if (objList.Count == 0) { return; }
13 //生成EXCEL的HTML
14 StringBuilder excelStr = new StringBuilder(objList.Count * columnInfo.Count);
15
16 Type myType = objList[0].GetType();
17 //根据反射从传递进来的属性名信息得到要显示的属性
18 List<PropertyInfo> myPro = new List<PropertyInfo>();
19 foreach (string cName in columnInfo.Keys)
20 {
21 PropertyInfo p = myType.GetProperty(cName);
22 if (p != null)
23 {
24 myPro.Add(p);
25 excelStr.Append(columnInfo[cName]).Append("\t");
26 }
27 }
28 //如果没有找到可用的属性则结束
29 if (myPro.Count == 0) { return; }
30 excelStr.Append("\n");
31
32 foreach (T obj in objList)
33 {
34 foreach (PropertyInfo p in myPro)
35 {
36 excelStr.Append(p.GetValue(obj, null)).Append("\t");
37 }
38 excelStr.Append("\n");
39 }
40
41 //输出EXCEL
42 HttpResponse rs = System.Web.HttpContext.Current.Response;
43 rs.ContentEncoding = System.Text.Encoding.GetEncoding("GB2312");
44 rs.AppendHeader("Content-Disposition", "attachment;filename=" + FileName);
45 rs.ContentType = "application/ms-excel";
46 rs.Write(excelStr);
47 rs.End();
48 }
49 }
在实例化StringBuilder excelStr = new StringBuilder(objList.Count * columnInfo.Count);时候预分配开始大小,这样能更好的使用StringBuilder。至此,改进完成。
另外,如果您觉得反射会影响性能,那么可以改成表达式树的方式,或者使用limit。