测试环境:windows2003,MSSQL2000,IIS6.0,Asp.net2.0
需求:根据用户查询条件将满足要求的信息查询出来生成xls文件供用户下载.
1.使用Excel组件生成xls文件,不采用CSV文件(逗号分割文件).
2.下载操作时采用ajax方式进行,需要提示界面.
3.需要将下载后的文件删除.
设计的过程:
1.使用Excel软件设计一个空的模版文件(temp.xls)被用.
2.每次生成时,将temp.xls拷贝一份到临时目录,文件名称随机生成(ran.xls).
3.根据用户提交的条件将数据填充到ran.xls文件中.
4.下载ran.xls文件.
5.将临时目录中的ran.xls文件删除.
分析:
A.过期文件删除处理:
针对第5点,一般可以使用定时清理程序根据文件生成时间进行清理(比方一天前的文件则删除掉),不过考虑到生成的xls文件一般在100KB以内,而且这个导出操作不很频繁(只对VIP用户提供),那么这里采用下载后马上删除的方式来实现,开始尝试了如下下载代码:
//=======注意***以下代码不能得到预期效果==============
Response.TransmitFile(filename);
//重载WriteFile(filename,true)是把整个文件读入内存再输出,跟下面的BineryWrite(byte[])表现是一样的
//Response.WriteFile(filename);//部分缓存文件
//即使加了下面两句,程序的表现也是不稳定的
//Response.Flush();
//Response.Close();
System.IO.File.Delete(filename);
//=========End==========
参考(http://forums.asp.net/t/1031479.aspx,)提到的在每次使用TransmitFile发送文件时,调用一次过期文件删除进程(按文件生成时间).考虑到TransmitFile能稳定的发送大文件,实际应用时建议按这种方式实施.
针对当前项目,这里采用一次性读入内存再逐步输出的方式进行
//===============目前采用的方式============
Response.ContentType = "application/ms-excel";
Response.AddHeader("Content-Disposition", "attachment;filename=" + HttpUtility.UrlEncode(savefilename, Encoding.GetEncoding("utf-8")));
Response.BinaryWrite(File.ReadAllBytes(filename));//直接将输出写入
Response.Flush();
System.IO.File.Delete(filename);
B.使用ajax方式下载问题:
用户点下载按钮后将下载按钮设置为不可用(防止多次点击),并且显示一个处理中的gif动画.
首先碰到的问题是使用ajax方式无法弹出文件下载对话框,后来找到使用iframe的形式来实现,相关的jquery代码大概是这样的,$("#frmDownload").attr("src","download.ashx?条件");.
起初是将生成Excel表格,发送下载文件做在一个过程里面的,使用iframe方式时,从点了下载按钮后到弹出文件下载对话框为止,对应服务器的处理时间,这段时间内客户端需要将下载按钮butOK禁用,然后显示一个gif,当弹出下载对话框时,再将按钮恢复并且隐藏gif动画,尝试了在ifrmae.onload里捕获下载框弹出事件,结果发现iframe里如果加载的是html页面时会触发,但是当是文件下载时就没有触发,估计是文件下载时IE解释器不生成DOM模型,当然也就没有JS事件了.
目前解决方法:
把Excel生成跟服务器发出下信号(指上面下载代码那一部分)分开,即首先客户端发出请求,服务器收到后生成excel文件然后返回一段json格式的信息(成功与否,以及excel文件地址),客户端收到成功信息后隐藏gif动画,恢复按钮可用,然后设置iframe.src="下载地址",这样一来就解决了上面的问题.
--另外可以考虑,照样使用iframe.onload事件来处理excel生成结束事件(隐藏动画,恢复按钮),不过服务器应该返回是一个包含<iframe src="下载地址" ></firame>的页面,即iframe里再套个iframe, 这个没测试过想来应该可性,这样就不需要将excel处理跟下载分成两步了.
参考代码如下:
///****************ajax下载过程*************************
$("#butOK").click(function(){
$("#butOK").attr("disabled",true);
$("#tip").html("<img src='/js/loading.gif'/>正在处理请求...");
$.getJSON("download.ashx?m=1",function(data){
//这里是演示,实际应用应该根据data里的信息来决定下一步操作
$("#frmDownload").attr("src","download.ashx?m=2");//附加到iframe下载
$("#butOK").attr("disabled","");
$("#tip").html("");
});
});
//********************代码结束******************
C.Excel报表生成问题:
首先这个报表包括表头(第一行是神鹰医药招商网--代理信息目录,第二行是http://www.gyzs.net/ 导出时间:******),第三行是字段名称,而且表格使用了样式以便看上去更加美观,这样一来使用OLEDB方式的Insert语句(按数据库方式处理)就会出错,Insert好象将第一行是字段,第二行就是数据行了,也不知道有没有办法解决.
另外朋友提供了一个服务端导出xml格式的数据,然后在客户端使用VBA(Js操作Excel组件)生成表格的方式,可以避免在服务器上装Excel程序,配置权限,Excel注销(回收内参)等一系列问题,不过需要对Excel进行格式化(能否下在服务器xls模版再进行操作?),用户当前电脑Excel版本是否会对Js编写有要求,在Js中使用Excel组件是否有权限以及安全方面的问题等, 多需要逐一测试.
目前解决方法:
使用Miscrosoft Excel 10.0 Object Library (Office 2002)组件,(需要在服务器上安装并配置权限)
//*****************写excel文件参考代码**********************
System.Data.DataTable dt = LoadData();//加载符合条件的数据
Object miss = System.Reflection.Missing.Value;
Excel.Application ExcelApp = new Excel.ApplicationClass();
Excel.Workbooks wbooks = (Excel.Workbooks)ExcelApp.Workbooks;
Excel.Workbook wbook = wbooks.Open(Path, miss, miss, miss, miss, miss, miss, miss, miss, miss, miss, miss, miss, miss, miss);
//获得第一个工作薄
Excel.Worksheet wsheet = (Excel.Worksheet)wbook.Worksheets.get_Item(1);
ExcelApp.Visible = false;
ExcelApp.DisplayAlerts = false;
wsheet.Cells[2, 1] = string.Format("信息来源:神鹰医药网〖http://www.gyzs.net/〗 导出时间:{0}", DateTime.Now.ToString("yyyy-MM-dd"));
//填充具体的单元格
//逐个填充
//for (int i = 0; i < dt.Rows.Count; i++)
//{
// for (int j = 1; j <= 6; j++)
// {
// wsheet.Cells[i + 4, j] = dt.Rows[i][j - 1];
// }
//}
//使用Rang填充
object[,] arr = new object[dt.Rows.Count, dt.Columns.Count];
for (int iRow = 0; iRow < dt.Rows.Count; iRow++)
{
for (int iCol = 0; iCol < dt.Columns.Count; iCol++)
{
arr[iRow, iCol] = dt.Rows[iRow][iCol];
}
}
Excel.Range range = wsheet.get_Range(wsheet.Cells[4, 1], wsheet.Cells[dt.Rows.Count + 4 - 1, dt.Columns.Count]);
range.set_Value(miss, arr);
ReleaseObject(range);
//填充结束
wbook.Close(true, miss, miss);
ExcelApp.Quit();
ReleaseObject(wsheet);
ReleaseObject(wbook);
ReleaseObject(wbooks);
ReleaseObject(ExcelApp);
GC.Collect();
GC.WaitForPendingFinalizers();
//*******************结束***********************
说明:Excel 首先是Applicate对象,代表了Excel应用程序,一个Applicate对应多个WorkBook(集合WorkBooks),一个WorkBook包含多张WorkSheet(工作表),每张表包括多个Cell(单元格式), 上面代码注释了逐个单元格填充数据的方式,目前使用的是按Range按区域(如A1-C3有9个单元格A1表示左上角的哪个,
上面使用Get_Range(Cell[LRow,LCol],Cell[RRow,RCol])获取需要填充的区域,而Object[,] arr是一个二纬数组,这些数据将被填充到这个区域(Range)里. 使用区域填充比逐个填充速度要快,3-4倍(我这里测试的信息)
附加部分:在客户端使用ActiveX方式将Table内容导入到Excel表格里
//*******需要权限,并且用户安装了合适版本的office
// 指定页面区域“单元格”内容导入Excel
function CellAreaExcel(objid) {
var oXL = null;
var oWB = null;
var oSheet = null;
try {
oXL = new ActiveXObject("Excel.Application");
} catch (e) {
alert("Excel对象无法创建");
return;
}
try {
oWB = oXL.Workbooks.Add();
oSheet = oWB.ActiveSheet;
oSheet.Cells.NumberFormatLocal = "@";
var Lenr = gridlist.rows.length;
for (var i = 0; i < Lenr; i++) {
var Lenc = $id(objid).rows(i).cells.length;
for (var j = 0; j < Lenc; j++) {
if (i == 0) {
oSheet.Cells(i + 1, j + 1).Font.Bold = true;
oSheet.Cells(i + 1, j + 1).ColumnWidth = $id(objid).rows(i).cells(j).clientWidth / 7;
}
oSheet.Cells(i + 1, j + 1).value = $id(objid).rows(i).cells(j).innerText;
}
}
oXL.Visible = true;
} catch (e) {
alert("导出到excel出错");
}
oXL.Quit();
oXL = null;
oWB = null;
oSheet = null;
idTmr = window.setInterval("Cleanup();", 1);
}