最近研究了一下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();
就当抛砖引玉了~