以前开发过一个用AspNET上传大文件的代码由于是在NET下开发的发现不能在NET下运行前几天将这个问题解决了现在把整个开发过程整理一下供大家参考由于内容较多所以我打算分成六篇来写第一篇写一下需要解决的问题和大致解决思路再用四篇具体写一下开发中需要解决的四个主要问题最后再写一个组件的使用方法
大家可以点击这里下载NET的源代码
问题提出
使用AspNET在上传文件时IIS接收到请求内容后发送到AspNET的管道中然后AspNET的ISAPI将读取的内容是全部装入服务器内存(NET在请求内容较大时会存放到临时文件中见后面的开发说明)在上传文件时消耗服务器资源较多所以AspNET对上传文件的大小会有限制且不能提供上传进度显示而我们在实际应用中往往会上传较大的文件的现象这时可以通过修改配置文件将最大将求长度配置设置得很大但这样服务器将会消耗大量内存
为此如果可以在IIS将客户端发送的内容发送到AspNET的管道时随着服务器接收内容的同时将文件内容写到服务器磁盘中然后在AspNET的管道中只放入请求的对应的Form内容这样可以避开AspNET上传文件的大小限制同时提供服务器接收进度显示
解决方案分析
AspNET提供了HttpModule功能提供了让用户处理客户请求的手段要实现HttpModule功能用户只需实现SystemWebIHttpModule接口SystemWebIHttpModule定义了两个方法void Init (HttpApplication context)和void Dispose ()Init方法给我们提供了一个类型为HttpApplication的参数这个HttpApplication类型中给我们提供了处理AspNET页面生命周期中发生的各种事件(具体请参照MSDN)我们只需要在最早的事件BeginRequest中处理数据接收就可以避开AspNET的最大请求长度限制
为了不改变AspNET对页面处理逻辑我们需要在提供文件内容后将其它请求内容正确封装给AspNET为此我们需要分析客户端发送到服务器端内容的具体格式我们可以通过其它工具提供这些内容这些内容大致如下
dedc
ContentDisposition: formdata; name=UploadID
dedebfaadaefdf
dedc
ContentDisposition: formdata; name=__VIEWSTATE
/wEPDwULLTIwODEOTkzNTUPZBYCAgMPFgIeBVuYRcGUFEbHRpcGFydCmbJtLWRhdGFkZAQpcdkiSZ/MOeenKOTGFxMk
dedc
ContentDisposition: formdata; name=TextBox
dedc
ContentDisposition: formdata; name=TextBox
dedc
ContentDisposition: formdata; name=file; filename=C:\Documents and Settings\Administrator\??????\componentartwebuirar
ContentType: application/xrarcompressed
Rar! ??s这里我把文件具体内容去掉了我们提取文件后封送给AspNET也就是去掉这部分后的内容
dedc
ContentDisposition: formdata; name=Button
开始上传
dedc
ContentDisposition: formdata; name=__EVENTVALIDATION
/wEWBALwVGAuzRsusGAuzRtkMAoznisYGMJIxZyapejeOQaIIwcOkPrWKnE=
dedc
请注意红色注释部分那里原来是上传的文件内容我们封送给AspNET就应当时去掉文件内容后内容这样页面控件的值页面事件就会正确的发送到AspNET页面只是页面不能正确获取上传文件的内容
这里又出现了三个问题一是我们如何获取文件内容如何确定那部分内容是上传的文件内容而不是页面其它控件内容二是我们如何在提取文件内容后让AspNET页面正确获取文件三是我们如何将改写后的内容封送给AspNET让它感觉不到我们提取了文件内容
只要解决了上面三个问题那大文件上传的问题就解决了最后就是提供进度显示由于内容较多我分四部分来介绍这三个问题和进度显示