SQL2005CLR函数扩展-数据导出的实现详解
前端之家收集整理的这篇文章主要介绍了
SQL2005CLR函数扩展-数据导出的实现详解,
前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
sqlServer数据导出到excel有很多种方法,比如dts、ssis、还可以用sql语句调用openrowset。我们这里开拓思路,用CLR来生成Excel文件,并且会考虑一些方便操作的细节。
下面我先演示一下我实现的效果,先看测试语句
--------------------------------------------------------------------------------
<FONT style="COLOR: #ff0000">exec BulkCopyToXls 'select
from testTable','d:/test','testTable',- 1
/ 开始导出数据
文件 d:/test/testTable.0.xls,共65534条,大小20,450,868 字节
文件 d:/test/testTable.1.xls,大小 20,101,773 字节
文件 d:/test/testTable.2.xls,040,589 字节
文件 d:/test/testTable.3.xls,大小 19,948,925 字节
文件 d:/test/testTable.4.xls,080,974 字节
文件 d:/test/testTable.5.xls,056,737 字节
文件 d:/test/testTable.6.xls,590,933 字节
文件 d:/test/testTable.7.xls,共26002条,大小 8,419,533 字节
导出数据完成
-------
共484740条数据,耗时 23812ms
*/
--------------------------------------------------------------------------------
上面的BulkCopyToXls存储过程是自定的CLR存储过程。他有四个参数:
第一个是
sql语句用来
获取数据集
第二个是
文件保存的路径
第三个是结果集的名字,我们用它来给
文件命名
第四个是限制单个
文件可以保存多少条记录,小于等于0表示最多65534条。 前三个参数没有什么特别,最后一个参数的设置可以让一个数据集分多个excel
文件保存。比如传统excel的最大容量是65535条数据。我们这里参数设置为-1就表示导出达到这个数字之后
自动写下一个
文件。如果你设置了比如100,那么每导出100条就会
自动写下一个
文件。 另外每个
文件都可以
输出字段名作为表头,所以单个
文件最多容纳65534条数据。 用微软公开的biff8格式通过二进制流
生成excel,服务器无需安装excel组件,而且
性能上不会比
sql自带的
功能差,48万多条数据,150M,用了24秒完成。
--------------------------------------------------------------------------------
下面我们来看下CLR
代码。通过
sql语句
获取DataReader,然后分批用biff格式来写xls
文件。
--------------------------------------------------------------------------------
<div class="codetitle">
<a style="CURSOR: pointer" data="86161" class="copybut" id="copybut86161" onclick="doCopy('code86161')"> 代码如下:
<div class="codebody" id="code86161">
using System;
using System.Data;
using System.Data.
sqlClient;
using System.Data.
sqlTypes;
using Microsoft.
sqlServer.Server;
public partial class StoredProcedures
{
///
/// 导出数据
/// /// <param name="
sql">
/// <param name="savePath">
/// <param name="tableName">
/// <param name="maxRecordCount">
[Microsoft.
sqlServer.Server.
sqlProcedure ]
public static void BulkCopyToXls(
sqlString
sql,
sqlString savePath,
sqlString tableName,
sqlInt32 maxRecordCount)
{
if (
sql.IsNull || savePath.IsNull || tableName.IsNull)
{
sqlContext .Pipe.Send(" 输入信息不完整!" );
}
ushort _maxRecordCount = ushort .MaxValue-1; if (maxRecordCount.IsNull == false && maxRecordCount.Value < ushort .MaxValue&&maxRecordCount.Value>0)
_maxRecordCount = (ushort )maxRecordCount.Value; ExportXls(
sql.Value,savePath.Value,tableName.Value,_maxRecordCount);
} ///
/// 查询数据,生成文件
/// /// <param name="
sql">
/// <param name="savePath">
/// <param name="tableName">
/// <param name="maxRecordCount">
private static void ExportXls(string
sql,string savePath,string tableName,System.UInt16 maxRecordCount)
{ if (System.IO.Directory .Exists(savePath) == false )
{
System.IO.Directory .CreateDirectory(savePath);
} using (
sqlConnection conn = new
sqlConnection ("context connection=true" ))
{
conn.Open();
using (
sqlCommand command = conn.CreateCommand())
{
command.CommandText =
sql;
using (
sqlDataReader reader = command.ExecuteReader())
{
int i = 0;
int totalCount = 0;
int tick = System.Environment .TickCount;
sqlContext .Pipe.Send(" 开始导出数据" );
while (true )
{
string fileName = string .Format(@"{0}/{1}.{2}.xls",savePath,tableName,i++);
int iExp = Write(reader,maxRecordCount,fileName);
long size = new System.IO.FileInfo (fileName).Length;
totalCount += iExp;
sqlContext .Pipe.Send(string .Format("
文件{0},共{1} 条,大小{2} 字节",fileName,iExp,size.ToString("###,###" )));
if (iExp < maxRecordCount) break ;
}
tick = System.Environment .TickCount - tick;
sqlContext .Pipe.Send(" 导出数据完成" );
sqlContext .Pipe.Send("-------" );
sqlContext .Pipe.Send(string .Format(" 共{0} 条数据,耗时{1}ms",totalCount,tick));
}
}
}
}
///
/// 写单元格
/// /// <param name="writer">
/// <param name="obj">
/// <param name="x">
/// <param name="y">
private static void WriteObject(ExcelWriter writer,object obj,System.UInt16 x,System.UInt16 y)
{
string type = obj.GetType().Name.ToString();
switch (type)
{
case "
sqlBoolean" :
case "
sqlByte" :
case "
sqlDecimal" :
case "
sqlDouble" :
case "
sqlInt16" :
case "
sqlInt32" :
case "
sqlInt64" :
case "
sqlMoney" :
case "
sqlSingle" :
if (obj.ToString().ToLower() == "null" )
writer.WriteString(x,y,obj.ToString());
else
writer.WriteNumber(x,Convert .ToDouble(obj.ToString()));
break ;
default :
writer.WriteString(x,obj.ToString());
break ;
}
}
///
/// 写一批数据到一个excel 文件
/// /// <param name="reader">
/// <param name="count">
/// <param name="fileName">
///
private static int Write(
sqlDataReader reader,System.UInt16 count,string fileName)
{
int iExp = count;
ExcelWriter writer = new ExcelWriter (fileName);
writer.BeginWrite();
for (System.UInt16 j = 0; j < reader.FieldCount; j++)
{
writer.WriteString(0,j,reader.GetName(j));
}
for (System.UInt16 i = 1; i <= count; i++)
{
if (reader.Read() == false )
{
iExp = i-1;
break ;
}
for (System.UInt16 j = 0; j < reader.FieldCount; j++)
{
WriteObject(writer,reader.Get
sqlValue(j),i,j);
}
}
writer.EndWrite();
return iExp;
} ///
/// 写excel 的对象
/// public class ExcelWriter
{
System.IO.FileStream _wirter;
public ExcelWriter(string strPath)
{
_wirter = new System.IO.FileStream (strPath,System.IO.FileMode .OpenOrCreate);
}
///
/// 写入short 数组
/// /// <param name="values">
private void _writeFile(System.UInt16 [] values)
{
foreach (System.UInt16 v in values)
{
byte [] b = System.BitConverter .GetBytes(v);
_wirter.Write(b,b.Length);
}
}
///
/// 写文件头
/// public void BeginWrite()
{
_writeFile(new System.UInt16 [] { 0x809,8,0x10,0 });
}
///
/// 写文件尾
/// public void EndWrite()
{
_writeFile(new System.UInt16 [] { 0xa,0 });
_wirter.Close();
}
///
/// 写一个数字到单元格x,y
/// /// <param name="x">
/// <param name="y">
/// <param name="value">
public void WriteNumber(System.UInt16 x,System.UInt16 y,double value)
{
_writeFile(new System.UInt16 [] { 0x203,14,x,0 });
byte [] b = System.BitConverter .GetBytes(value);
_wirter.Write(b,b.Length);
}
///
/// 写一个字符到单元格x,y
/// /// <param name="x">
/// <param name="y">
/// <param name="value">
public void WriteString(System.UInt16 x,string value)
{
byte [] b = System.Text.Encoding .Default.GetBytes(value);
_writeFile(new System.UInt16 [] { 0x204,(System.UInt16 )(b.Length + 8),(System.UInt16 )b.Length });
_wirter.Write(b,b.Length);
}
}
};