最近研究了一下IE自带的一些Activex控件,可以比较简单的实现断点续传功能
不过这种方式不推荐,因为安全性较低,而且需要修改客户端注册表,调低ie安全配置
还有就是我比较懒,只打算写个思路和几个关键的api功能
断点续传的基本思路如下:
1、客户端计算文件md5值,发送到服务器端校验;
2、服务器端根据客户端发送过来的md5值来确定文件是否存在,并返回文件大小,这样可以确定断点续传的文件开始位置;
3、客户端根据服务器返回的文件开始位置,开始读取文件块,每读取一块就发送给服务器,发送成功则读取下一块;
启用ie的ADODB.Stream
读取文件可以用ADODB.Stream,支持文件指针,可以从指定位置开始读取文件
但是使用的话,需要先修改注册表,找到HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Internet Explorer\ActiveX Compatibility\{00000566-0000-0010-8000-00AA006D2EA4}
在右窗格中,双击“Compatibility Flags”,在“编辑 DWORD 值”对话框中,确保选中“十六进制”选项,在“数值数据”框中键入 0,然后单击“确定”。
读取文件计算md5
计算文件可以用CAPICOM.HashedData,如果文件太大可以分次读取,挺简单的,直接上代码
function getMD5(){ var stream = new ActiveXObject("ADODB.Stream"); var hashed = new ActiveXObject("CAPICOM.HashedData"); if (stream == null || hashed == null){ alert("not support"); } else{ hashed.Algorithm = 3;//hash类型MD5 stream.Type = 1;//指定读取方式为二进制 stream.Open(); stream.LoadFromFile("E:\\TDownloads\\Http.rar"); stream.Position = 0;//读取文件的位置 var filesize = stream.Size;//文件大小 var blocksize = 10240;//读取块大小 var readCount = 0;//读取计数 var totalCount = Math.ceil(filesize/blocksize);//总共需要读取的次数 for (var i=0;i<totalCount;i++){ hashed.Hash(stream.Read(blocksize)); } stream.Close(); alert("MD5=" + hashed.Value); } }
stream.Read(blocksize)是每次读取blocksize大小的二进制数据,stream.Read()将一次性全部读取出来
发送文件碎片
现在客户端可以读取文件碎片,也可以计算md5了,就差异步发送文件碎片了
可以使用Msxml2.XMLHTTP发送二进制数据,js本身不支持发送二进制数据
如果一定要用js来发送文件碎片的话,可以变通的把二进制转换成base64字符串,然后服务器端再解密一次
function sendData(){ ... var data = stream.Read(blocksize);//读取出文件碎片 ... var xmlhttp = new ActiveXObject("Msxml2.XMLHTTP"); //这里的具体url可以自己定义 xmlhttp.Open("POST","uploadfile?filename=Http.rar&position=0&blocksize=10240&md5=md5",false); xmlhttp.setRequestHeader("Content-Length",blocksize); xmlhttp.send(data);//发送的二进制数据 alert(xmlhttp.responseText);//回执 }
以java为例,在服务器端接收的时候,url中的参数可以如下获取
String filename = req.getParameter("filename");
二进制数据可以直接以流的方式获取
HttpServletRequest.getInputStream();
就当抛砖引玉了~