xmlhttp 分块上传文件

编写思路:把本地文件在客户端通过base64编码以后发送目的地.
测试过程中,上传文件过大,导致超时不成功.
后来经过改善.把编码分段发送.测试20M成功

编写目的:在传统的解决方案里面,一次一次选取上传可以.但是在碰到把数据库里文件路径读出来,并把这些文件上传到一个地方的时候就比较麻烦.
如果得到路径一个一个去找到用ftp当然也是可以的,但每次找这些文件我看都会比较费时。这里编写这个主要就是为了通过数据库里的文件路径取得文件.把文件一次批量上传到一个地方.
其主要目的还是为了锻炼一下自己.

解决过程:起初试着用模拟键盘输入强行赋值给file控件用传统的方法上传。可是老碰到空值的情况,以至有很多文件没有发送出去。查阅一些资料,现在是把所有路径通过数据库取到然后写到一个js里。然后在前台用js读取这些路径,通过xmlhttp来发送文件。
因为ie不太喜欢xmlHttp.总认为他有恶意行为.所以老谈出提示警告.所以操作的时候不能使用web路径.只能用物理路径去访问它.
然后服务端有一文件来接收这些编码,并对其进行解码.所以我称之为"c/s". ^_^

目前很多代码还在完善中.
简单介绍一下:

aryFiles.push("c:\\aaa.zip") ;
aryFiles.push("c:\\bbb.exe") ;
这里为文件路径和文件.可为多个
以后这个路径也可以通过file控件在客户端取到

http:// www.xxx.com/xxx/xxx.asp
这个是目的地,可以改为自己想要的地址.

ado_stream.LoadFromFile(server.mappath(".") &"/"& + str_filename)
server.mappath(".") &"/"& + str_filename 这里是读取文件.
server.mappath(".") &"/"& 路径和存放路径一致

ado_stream.SaveToFile server.mappath(".") &"/"& str_filename,2
server.mappath(".") &"/"& 这个为存放文件的路径. str_filename 为文件名

这里读取和存放都是放在程序所放目录中.大家测试的时候也可以保持这样即可

把第一段代码放到本地(eg:c:\upload.htm)

把第二段代码放到服务器上,可以是本地服务器,可以是公网服务器.和上面的目的地保持一致
(eg:http://www.xxx.com/upload.asp or http:// localhost/www/upload.asp)

操作:找到第一段代码保存的地方。执行即可(eg:打开c盘执行upload.htm)

目前程序还在调试过程中,希望大家提出宝贵意见

客户端代码<html><head></head><body> <input type=button value="发送" /> <input type=button value="中断" /> <div id="ddd"width=300px></div> <br></br> <DIVid=div_message></DIV></body><script language=VBScript> Function bytes2BSTR(vIn) strReturn ="" For i = 1 To LenB(vIn) ThisCharCode = AscB(MidB(vIn,i,1)) IfThisCharCode < &H80 Then strReturn = strReturn &Chr(ThisCharCode) Else NextCharCode = AscB(MidB(vIn,i+1,1)) strReturn =strReturn & Chr(CLng(ThisCharCode) * &H100 +CInt(NextCharCode)) i = i + 1 End If Next bytes2BSTR = strReturn EndFunction</script><script language=javascript> var xmlhttp ; var ado_stream ; varmFileName, mPartStart, mPartID, mPartEnd ; var SendCount ; varBlockSize ; var Breaked ; var aryFiles ; BlockSize = 1024*100;//每次发送字节数 Breaked = false ; aryFiles = new Array() ; // 开始发送文件function BeginSendFiles() { initAryFiles() ; SendFile(aryFiles.pop()) ;} // 构造待发送文件的数组 function initAryFiles() { aryFiles.push("c:\\aaa.zip"); aryFiles.push("c:\\bbb.exe") ; //c:\\aaa.zip c:\\bbb.exe本地文件aryFiles.reverse() ;//文件名 } function SendFile(vFullPath) { // 空文件则不执行上传if (!vFullPath) { return ; } Breaked = false ; div_message.innerHTML ="" ; ado_stream = new ActiveXObject("ADODB.Stream"); // 读取文件的流ado_stream.Type = 1; ado_stream.Open();ado_stream.LoadFromFile(vFullPath); // 读取文件 ado_stream.position = 0 ;SendCount = Math.ceil(ado_stream.size/BlockSize) ; // 如果有余数则多发送一次 //alert(SendCount) ; var reg = /\b\w+.\w+$/gi mFileName =reg.exec(vFullPath) ; mPartStart = true ; mPartID = 1 ; mPartEnd =false ; SendData() ; } function SendData() { if (SendCount > 0) {var dom = new ActiveXObject("msxml2.DOMDocument"); // 发送的xml文件dom.async = false; dom.resolveExternals = false; // 构造xml文件头 var node =dom.createProcessingInstruction("xml","version='1.0'");dom.appendChild(node) ; node = null ; // 构造root节点 var root =dom.createElement("root"); dom.appendChild(root) ;dom.documentElement.setAttribute("xmlns:dt","urn:schemas-microsoft-com:datatypes"); // 构造保存二进制数据的节点updata node =dom.createElement("upData") ; node.dataType = "bin.base64" ; //bin。base64编码 var att = dom.createAttribute("FileName") ; // 文件名属性att.value = mFileName ; node.setAttributeNode(att) ; att = null ; varatt = dom.createAttribute("PartStart") ; // 分段开始标记 att.value =mPartStart ; node.setAttributeNode(att) ; att = null ; var att =dom.createAttribute("PartID") ; // 分段序号 att.value = mPartID ;node.setAttributeNode(att) ; att = null ; var att =dom.createAttribute("PartEnd") ; // 分段结束标记 att.value = mPartEnd ;node.setAttributeNode(att) ; att = null ; root.appendChild(node) ;node.nodeTypedValue = ado_stream.Read(BlockSize); // 节点数据从stream读取,固定长度node = null ; SendCount -= 1 ; xmlhttp = newActiveXObject("Microsoft.XMLHTTP");xmlhttp.open("POST","http://www.xxx.com/xxx/xxx.asp", false);//http://www.xxx.com/xxx/xxx.asp 为web路径上的文件 xmlhttp.onreadystatechange=CallBack ; xmlhttp.send(dom); mPartStart = false ; xmlhttp = null ; }else { ado_stream.Close(); ado_stream = null ; } } function CallBack(){ // 上传成功 if(xmlhttp.readystate == 4) { // 检查是否中断上传 if(Breaked) {return ; } if (SendCount > 0) { mPartID += 1 ; //div_message.innerHTML += ("<br>" + xmlhttp.ResponseText) ; var p= Math.floor((mPartID/(Math.ceil(ado_stream.size/BlockSize) + 1)) *100) ; // 计算进度百分比 ShowBar(p) ; var t = setTimeout("SendData();", 1) ; }else { // 传送完文件 //div_message.innerHTML += mFileName +"传送完毕!<br>" ; // 继续传下一文件 ShowBar(0) ; var cFile = aryFiles.pop(); SendFile(cFile) ; } } } function ShowBar(per) { // 进度条 ddd.innerHTML= "<table width='200' border=0 cellpadding='0' cellspacing='0'><tr><td bgcolor='#6699FF'><input type=button style='width:" + per + "% ; border:0px; background:#005599; color:#FFFFFF'value=" + per + "%> </td></tr></table>" ; }</script></html> [Ctrl+A 全部选择 提示:你可先修改部分代码,再按运行]

服务端代码<%@ LANGUAGE=VBScript%> <% Option Explicit Response.Expires = 0 ' 定义变量和对象。 dim ado_stream dim xml_dom dim xml_data dim str_filenamedim bol_PartStartdim int_PartIDdim bol_PartEnd ' 创建 Stream 对象 set ado_stream = Server.CreateObject("ADODB.Stream") ' 从Request对象创建 XMLDOM对象 set xml_dom = Server.CreateObject("MSXML2.DOMDocument") xml_dom.load(request) ' 读出包含二进制数据的节点 set xml_data = xml_dom.selectSingleNode("root/upData") str_filename = xml_data.getAttribute("FileName")bol_PartStart = CBool(xml_data.getAttribute("PartStart"))int_PartID = CInt(xml_data.getAttribute("PartID"))bol_PartEnd = CBool(xml_data.getAttribute("PartEnd"))' 打开Stream对象,把数据存入其中 ado_stream.Type = 1 ' 1=adTypeBinary ado_stream.open if not bol_PartStart then ado_stream.LoadFromFile(server.mappath(".") &"/"& + str_filename) ' 读取文件 ado_stream.position = ado_stream.size end ifado_stream.Write xml_data.nodeTypedValue ' 文件存盘 ado_stream.SaveToFile server.mappath(".") &"/"& str_filename,2 '保存文件 2=adSaveCreateOverWrite ado_stream.close ' 释放资源 set ado_stream = Nothing set xml_dom = Nothing ' 向浏览器返回信息 Response.Write "Upload successful!"& str_filename & int_PartID & bol_PartStart%>
 [Ctrl+A 全部选择 提示:你可先修改部分代码,再按运行]

另外:此文的目的并不是说这个程序能给朋友们解决什么问题,主要目的还是为了和大家交流一些心得:
碰到问题->发现问题->解决问题.
希望大家都不要成为一个标准的代码机器人。
多学、多看、多思考、多实践。
没有什么是不能解决的。

时间: 2024-11-02 23:02:25

xmlhttp 分块上传文件的相关文章

利用Xmlhttp 分块上传文件

xml|上传 编写思路:把本地文件在客户端通过base64编码以后发送目的地.测试过程中,上传文件过大,导致超时不成功.后来经过改善.把编码分段发送.测试20M成功 编写目的:在传统的解决方案里面,一次一次选取上传可以.但是在碰到把数据库里文件路径读出来,并把这些文件上传到一个地方的时候就比较麻烦.如果得到路径一个一个去找到用ftp当然也是可以的,但每次找这些文件我看都会比较费时.这里编写这个主要就是为了通过数据库里的文件路径取得文件.把文件一次批量上传到一个地方.其主要目的还是为了锻炼一下自己

asp.net-C# Activx 上传分块上传大文件时能由Activx来完成合并分块的工作吗

问题描述 C# Activx 上传分块上传大文件时能由Activx来完成合并分块的工作吗 我用C#做activx上传大文件,采用http分块上传的思想,可是服务器端的人说,分块和拼接的工作都应由activx完成,上传完了给服务器一个完成的信号,服务器端只负责把拼接的文件编入数据库和移动到合适的位置,可是网上的好多资料都是拼接工作在服务器端完成,Activx真的能实现远程拼接工作吗,如果能,能否给一些资料,谢谢 解决方案 服务器端的人说的不对,ActiveX只需要分块就可以了.再拼接上显然是服务器

c#-C#大文件分块上传原理问题

问题描述 C#大文件分块上传原理问题 我对大文件分块上传的概念是,通过offset每次读取1M文件流,再通过http发送Content-Range和这1M文件流,换句话说分块上传是不是需要每次都发送请求头?我看见到有些代码,只发送一次文件头,然后不断发送文件流,虽然没成功,但是留下了疑惑 解决方案 通过 HTTP 进行大数据传输,不是好的选择. 如果一定要用 HTTP,传输时 HTTP 协议头肯定是不可少的.至于其中的数据,由于每次 HTTP 都是一次 Socket 连接.数据传输.断开的过程,

有谁用VB6做过HTTP上传文件吗,用Microsoft.XMLHTTP或者其它组件,上传到.NET或者JAVA的WEB服务器

问题描述 问题的提出:客户打开网页,已经知道客户本地硬盘上的文件路径和目录,在客户不打开文件选择窗口的情况下,把客户的文件上传到J2EE的WEB服务器,采用JAVA的COS组件来存取文件.初步的解决办法:用VB6做一个OCX控件,放在网页里面,把文件地址和WEB服务器上传路径传给这个控件,然后调用这个控件的上传方法,上传文件,找了很多资料后,只是做到了可以上传文本文件,二进制文件没有搞定.然后做了一个网页,提交一个FORM(method="post"enctype="mult

ASP跨服务器上传文件完美解决

服务器|解决|上传 如果你有两个空间,一个大而慢,另外一个小而快,或者其中一个不支持FSO,那么跨服务器上传文件这个问题就摆在你面前了,下面就是我在解决ylog.net里面的跨服务器上传文件的日记,这个问题看似简单,里面的细节问题却是非常有趣 前提条件,空间都必须支持ASP,上传文件的服务器支持FSO,下面的叙述中,diygame.com为存上传文件的服务器,ylog.net为网站服务器,即显示用户界面的服务器... 实现功能,文件上传,上传后在网页的文本区域自动加上对于图片的UBB码,为了清楚

XML技术上传文件-转贴

xml|上传                         XML技术上传文件 概述本文讲解了一个使用XML技术上传文件的例子,使用该方法没有传统方法中的种种限制. 这个例子讲述了如何使用MSXML3.0和ADO Stream对象来实现这种新的上传方法.好处有很多,比如,不需要专用的上传组件. 引言为了在HTML网页中获得上传功能,在客户端我们可以使用如下格式的FORM: <FORM NAME="myForm"ACTION="TargetURL.asp"EN

Js+php实现异步拖拽上传文件

         本文给大家分享的是使用php结合js实现异步拖拽上传文件的代码,及示例,有需要的小伙伴可以参考下.              异步拖拽上传文件--小实例 upload.html ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53

azure- 有关blob上传文件获得上传进度的问题

问题描述 有关blob上传文件获得上传进度的问题 大神好! ICloudBlob blockBlob = container.GetBlockBlobReference(fileblobname); await blockBlob.UploadFromFileAsync(file); 有没有现成的api可以获得上传进度, 如果没有的话,可以不可以提供一个思路?? 解决方案 Hi, 据我所知目前没有这样的API,不过我们可以尝试通过将文件分块的方式,然后对每一块的上传做一次监督,这样上传的文件和总

activexobject-ActiveXObject获取上传文件的二进制流

问题描述 ActiveXObject获取上传文件的二进制流 支持W3C File的浏览器,都可以使用FileReader对象获取上传文件的二进制流. 但是IE不支持,IE11都不支持. @showbo 感谢大虾帮我解决了问题.但是IE是个头疼的问题. 再次请教各位大虾,有什么能让IE获取上传文件二进制流的方法么? 不能使用flash,因为面向的用户特殊,他们基本不会电脑,更别说安装flash. IE获取不到ActiveXObject对象,我可以写bat文件,让用户下载后执行bat后再进行上传文件