本文所演示的导入和导出方法并未使用Excel COM组件,而是使用原始的ASP.NET OleDb驱动程序,这使得ASP.NET应用程序即使部署在未安装Excel的服务器上也同样能支持导入和导出校准的Excel格式文件。
[@more@]
本文所演示的导入和导出方法并未使用Excel COM组件,而是使用原始的ASP.NET OleDb驱动程序,这使得ASP.NET应用程序即使部署在未安装Excel的服务器上也同样能支持导入和导出校准的Excel格式文件。
一、导出
网上有不少关于ASP.NET应用程序导出Excel文件的方法,大多都是设置Response对象的Content-Type为“application/ms-excel”后,以字符串形式输出数据供用户下载(可参考:http://blog.csdn.net/jilm168/archive/2007/11/06/1869118.aspx)。
虽说这种方式确实能实现导出Excel文件的目的,但其导出的Excel文件是不标准的。如果你用记事本打开使用此方式导出的文件会发现该文件是一个纯文本文件,这个倒是次要的,因为强大的Excel绝对有能力识别并显示这种“伪”Excel文件,但如果我们的应用程序要支持导入的话,那么这一文件就无法被程序正确识别了。
为了导出标准的Excel文件,我们可以首先准备好一个标准Excel文件做为导出模板,导出数据时先创建此模板的一个副本,使用OleDb驱动程序向副本中写入数据后,将此副本的二进制字节流输出至客户端供用户下载,最后再删除此副本。
如何制做一个Excel文件模板呢?你可以在安装有Excel的机器上新建一个“Microsoft Excel 工作表”,在工作表的第一行设置好需要的字段,然后选中整个表,点右键选择“设置单元格格式”,在“数字”标签的“分类”列表选择“文本”,点击“确定” 并保存文件。将单元格格式设置为“文本”主要是为了防止日后导入数据时某些数据格式无法被识别。
有了这个模板,我们就可以开始写代码导出数据了。本示例所使用模板文件位于站点根目录,名为“考生成绩信息表demo.xls”,共包含三个列:“准考证号”、“姓名”。
excel_template
private void Excel(DataTable dt, string FileName)
{
// 根据模板文件创建副本
string a = Guid.NewGuid().ToString();
string filePath = Server.MapPath("../xls/" + Guid.NewGuid().ToString() + ".xls");
File.Copy(Server.MapPath("../xlsDemo/考生成绩信息表demo.xls"), filePath);
// 使用OleDb驱动程序连接到副本
OleDbConnection conn = new OleDbConnection("Provider=Microsoft.Jet.OLEDB.4.0;Data Source=" + filePath + ";Extended Properties=Excel 8.0;");
using (conn)
{
conn.Open();
// 增加记录
for (int i = 0; i < dt.Rows.Count; i++)
{
OleDbCommand cmd = new OleDbCommand("INSERT INTO [Sheet1$]([准考证号], [姓名]) VALUES('" + dt.Rows[i][0].ToString() + "', '" + dt.Rows[i][1].ToString() + "')", conn);
cmd.ExecuteNonQuery();
}
}
// 输出副本的二进制字节流
Response.ContentType = "application/ms-excel";
Response.AppendHeader("Content-Disposition", "attachment;filename=" + Server.UrlEncode(FileName));
Response.Charset = "GB2312";
Response.BinaryWrite(File.ReadAllBytes(filePath));
// 删除副本
File.Delete(filePath);
}
二、导入
相对于导出,导入部分的实现代码要简单的多:
string f_name = Server.MapPath("../xls/");
string sql = "";
string cs = "Provider=Microsoft.Jet.OLEDB.4.0;Data Source=" + f_name + ";Excel 8.0;HDR=Yes;IMEX=1";
OleDbConnection CNN = new OleDbConnection(cs);
CNN.Open();
string ss = "select * from [Sheet1$] ";
OleDbDataAdapter oda = new OleDbDataAdapter(ss,CNN);
DataSet ds = new DataSet();
oda.Fill(ds);
oda.Dispose();
for (int i = 0; i < ds.Tables[0].Rows.Count; i++)
{
sql="执行语句"
b = db.db.ExecuteSql1(sql);
}
if (b)
Maticsoft.Common.MessageBox.Show(Page, "导入完成!");
else
Maticsoft.Common.MessageBox.Show(Page, "导入失败!");
导入时我们依旧使用OleDb驱动程序来访问Excel文件。再次提醒读者,此种方式只支持导入标准的Excel文件,这也是上面导出部分使用预先准备好的Excel模板文件的原因。
另外,以上演示仅针对Excel 2000-2003格式的文件,如果您要导入或导出的是Excel 2007文件(*.xlsx),那么您需要将连接字符串改为(可参考:http://www.connectionstrings.com/):
Provider=Microsoft.ACE.OLEDB.12.0;Data Source=info.xlsx;Extended Properties="Excel 12.0 Xml;HDR=YES";
Provider=Microsoft.ACE.OLEDB.12.0;Data Source=info.xlsx;Extended Properties="Excel 12.0 Xml;HDR=YES";
-----经过实践,上面的方法是可行,但有个问题:执行效率很低,超过255行的数据导出时,特别的慢(在网上查得是MS的限制).解决方法是:将INSERT部分不使用参数模式,而直接使用串拼接成INSERT的SQL语句,执行起来速度很快,导出时也很快,超过255时也很快,我仅测试过用SQL串语句生成并导出2000条,文件大小有3M,速度是不慢的,完成可以接受.而使用原方法,在256条时就感觉到慢了.
另外:OLEDB模式下可以先用CREATE TABLE生成SHEET1$这种EXCEL工作表,然后再INSERT数据,证实是可行的.这样模板文件就只要一个空白的EXCEL文档了.灵活性高很多.仍然要感谢原作者提供的方法.解决了我工作中的问题:)3Q