本文通过一个实例概要讲解如何在asp.net程序中配合sql server2000进行word文件的存储和调用过程(没有使用vba )。
(1) 建立数据库
首先,我们在数据库中建立一个表,表中有三个字段,filename(varchar,50),posttime(datetime,8), filecontent(image,16),分别存储文件名称,上传时间和word文件的具体内容,其中filename为主键。具体的sql脚本如下:
create table [dbo].[word] (
[filename] [varchar] (50) collate chinese_prc_ci_as not null ,
[posttime] [datetime] not null ,
[filecontent] [image] not null
) on [primary] textimage_on [primary]
(2) 上传并存储word文件
在vs.net中建立一个asp.net web应用程序,在界面内加入如下控件
控件类型 | id | text | 说明 |
label | label1 | 请输入文档的标题 | |
label | label2 | 请选择具体文档 | |
file field | file1 | 上传控件(要将此html控件转化为服务器控件) | |
textbox | name_textbox | 用于录入文档标题 | |
button | btn_ok | 上传文件 | |
button | btn_get | 读取文件 | |
hyperlink | hyperlink1 | 打开 | 用于打开word文档 |
上传文件时首先通过上传控件找到所需上传的文件,然后获取文件的大小,最后以流的形式写入数据库,具体代码为:
private void btn_ok_click(object sender, system.eventargs e)
{
string name=name_textbox.text;
//接收上传文件
stream filestream=file1.postedfile.inputstream;
//获取上传文件字节的大小
int length=file1.postedfile.contentlength;
byte[] worddata=new byte[length];
//从流中读取字节并写入worddata
int n=filestream.read(worddata,0,length);
//获取当前时间
datetime time=datetime.now;
//连接数据库
sqlconnection conn=new sqlconnection();
conn.connectionstring="workstation id=tianchunzhu;packet size=4096;integrated security=sspi;data source=tianchunzhu;persist security info=false;initial catalog=test";
sqlcommand cmd=new sqlcommand();
cmd.connection=conn;
cmd.commandtext="insert into word (filename,posttime,filecontent) values (@filename,@posttime,@filecontent)";
sqlparameter nameparam=new sqlparameter("@filename",system.data.sqldbtype.varchar,50);
nameparam.value=name;
cmd.parameters.add(nameparam);
sqlparameter timeparam=new sqlparameter("@posttime",system.data.sqldbtype.datetime,8);
timeparam.value=time;
cmd.parameters.add(timeparam);
//添加word文件
sqlparameter contentparam=new sqlparameter("@filecontent",system.data.sqldbtype.image); ①//见本段最后注解
contentparam.value=worddata;
cmd.parameters.add(contentparam);
conn.open();
cmd.executenonquery();
conn.close();
}
注①:此处由于是image类型文件,事先可能无法预测文件的大小,因此可不必指定size参数。如果希望控制上传文件的大小则可以输入size参数。如指定1000,则上传时最大可以上传1k的word文档。
(3) 从数据库中读取数据并恢复为word文件
读取数据时先将数据从数据库中读入缓冲区,然后再从缓冲区写入最终文件。因此首先要开辟一个缓冲区并设定它的大小,每当缓冲区读满时就要将缓冲区内的数据写入文件,以清空缓冲区并继续向缓冲区读数据,直到最后一次将缓冲区内剩余的数据全部写入文件,新的word文档即可生成。
由于这一部分用到了字节流的输入输出操作,因此要引用system.io命名空间
下面是关于这一部分的完整代码:
private void btn_get_click(object sender, system.eventargs e)
{
//连接数据库
sqlconnection conn=new sqlconnection();
conn.connectionstring="workstation id=tianchunzhu;packet size=4096;integrated security=sspi;data source=tianchunzhu;persist security info=false;initial catalog=test";
sqlcommand cmd=new sqlcommand();
cmd.connection=conn;
//根据textbox中指定的文件名进行查找读取
cmd.commandtext="select filecontent from word where filename='"+name_textbox.text.tostring()+"'";
filestream fs;
binarywriter bw;
//设定允许读取到缓冲区的最大长度
int buffersize=100;
//要将字节流读入的缓冲区
byte[] outbyte=new byte[buffersize];
//用于记录已经读取的字节数
long reval;
//字段中的索引,从这里开始读取操作
long startindex;
//filestream对象将封装的文件的相对路径或绝对路径
string [email protected]"c:/worddata.doc";
conn.open();
sqldatareader reader;
reader=cmd.executereader();
while (reader.read())
{
fs=new filestream(filepath,filemode.openorcreate,fileaccess.write);
bw=new binarywriter(fs);
startindex=0;
//将字节流读入outbyte缓冲区中并返回读取的字节数
reval=reader.getbytes(0,startindex,outbyte,0,buffersize);
//当读取的字节流达到缓冲区允许的最大长度时要卸载缓冲区内的数据并将数据写入文件
while (reval==buffersize)
{
bw.write(outbyte);
bw.flush();
//重新设定开始读取的位置,并继续读取和写数据
startindex+=buffersize;
reval=reader.getbytes(0,startindex,outbyte,0,buffersize);
}
//将缓冲区内最后剩余的数据写入文件
bw.write(outbyte,0,(int)reval-1);
bw.flush();
bw.close();
fs.close();
}
reader.close();
conn.close();
}
此时将按照filepath中指定的路径和名称重新生成word文档。可以在filepath中根据具体情况指定生成的word文档的名称和路径。
(4) 打开word文档
在打开word文档这一部分暂时并没有找到通过button按钮直接打开word的有效办法,但我们可以hyperlink控件,只要将hyperlink控件的navigateurl属性指向word文档的物理路径就可以了。
新闻热点
疑难解答
图片精选