断点续传原理:
现在有一个文件需要我们进行下载,当我们下载了一部分的时候,出现情况了,比如:电脑死机、没电、网络中断等等。 对于以上行为,如果“下载”的行为无法记录本次下载的一个进度。那么,当我们再次下载这个文件也就只能从头来过。
所以,要实现让一种断开的行为“续”起来的目的,关键就在于要有“介质”能够记录和读取行为出现”中断”的这个节点的信息。
实际上这就是“断点续传”的基础原理,用大白话说就是:我们要在下载行为出现中断的时候,记录下中断的位置信息,在新的下载行为开始的时候,直接从记录的这个位置开始下载内容,而不再从头开始。
实现步骤:
1、当“上传(下载)的行为”出现中断,我们需要记录本次上传(下载)的位置(position)。
2、当“续”这一行为开始,我们直接跳转到postion处继续上传(下载)的行为。
实现方式:
这里多了一行 RANGE: bytes=2000070-
这一行的意思就是告诉服务器 abc.exe 穿上文件从 2000070 字节开始传,前面的字节不用传了。 服务器收到这个请求以后,返回的信息如下:
返回的信息增加了一行: Content-Range=bytes 2000070-106786027/106786028
返回的代码也变成 206 ,而不再是 200 了。知道了这些原理,就可以进行断点续传的编程了。
关键点:
1、用什么方法实现提交 RANGE: bytes=2000070-。
用最原始的 Socket 是肯定能完成的,那样太费事了,Java 的 net 包中已经提供了这种功能。代码如下:
从输入流中取出的字节流就是 abc.exe 文件从 2000070 开始的字节流。 其实断点续传用 Java 实现起来还是很简单的,接下来要做的事就是怎么保存获得的流到文件中去。
接下来要做的就是整合成一个完整的程序了。包括一系列的线程控制等等。
断点续传完整实现:
主要用了 6 个类,包括一个测试
现在有一个文件需要我们进行下载,当我们下载了一部分的时候,出现情况了,比如:电脑死机、没电、网络中断等等。 对于以上行为,如果“下载”的行为无法记录本次下载的一个进度。那么,当我们再次下载这个文件也就只能从头来过。
所以,要实现让一种断开的行为“续”起来的目的,关键就在于要有“介质”能够记录和读取行为出现”中断”的这个节点的信息。
实际上这就是“断点续传”的基础原理,用大白话说就是:我们要在下载行为出现中断的时候,记录下中断的位置信息,在新的下载行为开始的时候,直接从记录的这个位置开始下载内容,而不再从头开始。
实现步骤:
1、当“上传(下载)的行为”出现中断,我们需要记录本次上传(下载)的位置(position)。
2、当“续”这一行为开始,我们直接跳转到postion处继续上传(下载)的行为。
实现方式:
要从文件已经下载的地方开始继续下载,需要在客户端浏览器传给 Web 服务器的时候要多加一条信息(告诉服务器要从哪里开始下载),如下面要求从 2000070 字节开始。
GET /abc.exe HTTP/1.0
User-Agent: NetFox
RANGE: bytes=2000070-
Accept: text/html, image/gif, image/jpeg, *; q=.2, */*; q=.2
这里多了一行 RANGE: bytes=2000070-
这一行的意思就是告诉服务器 abc.exe 穿上文件从 2000070 字节开始传,前面的字节不用传了。 服务器收到这个请求以后,返回的信息如下:
206
Content-Length=106786028
Content-Range=bytes 2000070-106786027/106786028
Date=Mon, 30 Apr 2001 12:55:20 GMT
ETag=W/"02ca57e173c11:95b"
Content-Type=application/octet-stream
Server=Microsoft-IIS/5.0
Last-Modified=Mon, 30 Apr 2001 12:55:20 GMT
返回的信息增加了一行: Content-Range=bytes 2000070-106786027/106786028
返回的代码也变成 206 ,而不再是 200 了。知道了这些原理,就可以进行断点续传的编程了。
关键点:
1、用什么方法实现提交 RANGE: bytes=2000070-。
用最原始的 Socket 是肯定能完成的,那样太费事了,Java 的 net 包中已经提供了这种功能。代码如下:
URL url = new URL("http://www.sjtu.edu.cn/abc.exe");
HttpURLConnection httpConnection = (HttpURLConnection)url.openConnection();
// 设置 User-Agent
httpConnection.setRequestProperty("User-Agent","NetFox");
// 设置断点续传的开始位置
httpConnection.setRequestProperty("RANGE","bytes=2000070");
// 获得输入流
InputStream input = httpConnection.getInputStream();
从输入流中取出的字节流就是 abc.exe 文件从 2000070 开始的字节流。 其实断点续传用 Java 实现起来还是很简单的,接下来要做的事就是怎么保存获得的流到文件中去。
2、保存文件采用的方法。
我采用的是 IO 包中的 RandAccessFile 类。 API文档中对该类的说明:
此类的实例支持对随机访问文件的读取和写入。随机访问文件的行为类似存储在文件系统中的一个大型 byte 数组。
如果随机访问文件以读取/写入模式创建,则输出操作也可用;输出操作从文件指针开始写入字节,并随着对字节的写入而前移此文件指针。
写入隐含数组的当前末尾之后的输出操作导致该数组扩展。该文件指针可以通过 getFilePointer 方法读取,并通过 seek 方法设置。
操作相当简单,假设从 2000070 处开始保存文件,代码如下:
RandomAccess oSavedFile = new RandomAccessFile("down.zip","rw");
long nPos = 2000070;
// 定位文件指针到 nPos 位置
oSavedFile.seek(nPos);
byte[] b = new byte[1024];
int nRead;
// 从输入流中读入字节流,然后写到文件中
while((nRead=input.read(b,0,1024)) > 0)
{
oSavedFile.write(b,0,nRead);
}
接下来要做的就是整合成一个完整的程序了。包括一系列的线程控制等等。
断点续传完整实现:
主要用了 6 个类,包括一个测试