断点续传的原理

其实断点续传的原理很简单,就是在 Http 的请求上和一般的下载有所不同而已。

打个比方,浏览器请求服务器上的一个文时,所发出的请求如下:

假设服务器域名为 wwww.sjtu.edu.cn,文件名为 down.zip。




Http代码 ​HTTP断点续传_服务器​ ​HTTP断点续传_i++_02HTTP断点续传_服务器_03


  1. GET /down.zip HTTP/1.1  
  2. Accept: image/gif, image/x-xbitmap, image/jpeg, image/pjpeg, application/vnd.ms-   
  3. excel, application/msword, application/vnd.ms-powerpoint, */*   
  4. Accept-Language: zh-cn   
  5. Accept-Encoding: gzip, deflate   
  6. User-Agent: Mozilla/4.0 (compatible; MSIE 5.01; Windows NT 5.0)   
  7. Connection: Keep-Alive   
GET /down.zip HTTP/1.1
Accept: image/gif, image/x-xbitmap, image/jpeg, image/pjpeg, application/vnd.ms-
excel, application/msword, application/vnd.ms-powerpoint, */*
Accept-Language: zh-cn
Accept-Encoding: gzip, deflate
User-Agent: Mozilla/4.0 (compatible; MSIE 5.01; Windows NT 5.0)
Connection: Keep-Alive


 服务器收到请求后,按要求寻找请求的文件,提取文件的信息,然后返回给浏览器,返回信息如下:




Http代码 ​HTTP断点续传_服务器​ ​HTTP断点续传_i++_02HTTP断点续传_服务器_03


  1. 200  
  2. Content-Length=106786028  
  3. Accept-Ranges=bytes   
  4. Date=Mon, 30 Apr 2001 12:56:11 GMT   
  5. ETag=W/"02ca57e173c11:95b"  
  6. Content-Type=application/octet-stream   
  7. Server=Microsoft-IIS/5.0  
  8. Last-Modified=Mon, 30 Apr 2001 12:56:11 GMT   
200
Content-Length=106786028
Accept-Ranges=bytes
Date=Mon, 30 Apr 2001 12:56:11 GMT
ETag=W/"02ca57e173c11:95b"
Content-Type=application/octet-stream
Server=Microsoft-IIS/5.0
Last-Modified=Mon, 30 Apr 2001 12:56:11 GMT


 所谓断点续传,也就是要从文件已经下载的地方开始继续下载。所以在客户端浏览器传给 Web 服务器的时候要多加一条信息 -- 从哪里开始。

下面是用自己编的一个"浏览器"来传递请求信息给 Web 服务器,要求从 2000070 字节开始。




Http代码 ​HTTP断点续传_服务器​ ​HTTP断点续传_i++_02HTTP断点续传_服务器_03


  1. GET /down.zip HTTP/1.0  
  2. User-Agent: NetFox   
  3. RANGE: bytes=2000070-   
  4. Accept: text/html, image/gif, image/jpeg, *; q=.2, */*; q=.2   
GET /down.zip HTTP/1.0
User-Agent: NetFox
RANGE: bytes=2000070-
Accept: text/html, image/gif, image/jpeg, *; q=.2, */*; q=.2


 仔细看一下就会发现多了一行 RANGE: bytes=2000070-

这一行的意思就是告诉服务器 down.zip 这个文件从 2000070 字节开始传,前面的字节不用传了。

服务器收到这个请求以后,返回的信息如下:




Http代码 ​HTTP断点续传_服务器​ ​HTTP断点续传_i++_02HTTP断点续传_服务器_03


  1. 206  
  2. Content-Length=106786028  
  3. Content-Range=bytes 2000070-106786027/106786028  
  4. Date=Mon, 30 Apr 2001 12:55:20 GMT   
  5. ETag=W/"02ca57e173c11:95b"  
  6. Content-Type=application/octet-stream   
  7. Server=Microsoft-IIS/5.0  
  8. Last-Modified=Mon, 30 Apr 2001 12:55:20 GMT   
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


 和前面服务器返回的信息比较一下,就会发现增加了一行:




Http代码 ​HTTP断点续传_服务器​ ​HTTP断点续传_i++_02HTTP断点续传_服务器_03


  1. Content-Range=bytes 2000070-106786027/106786028   
Content-Range=bytes 2000070-106786027/106786028


 返回的代码也改为 206 了,而不再是 200 了。

知道了以上原理,就可以进行断点续传的编程了。

 

Java 实现断点续传的关键几点

(1) 用什么方法实现提交 RANGE: bytes=2000070-。

当然用最原始的 Socket 是肯定能完成的,不过那样太费事了,其实 Java 的 net 包中提供了这种功能。代码如下:




Java代码 ​HTTP断点续传_服务器​ ​HTTP断点续传_i++_02HTTP断点续传_服务器_03


  1. URL url = new URL("http://www.sjtu.edu.cn/down.zip");   
  2. HttpURLConnection httpConnection = (HttpURLConnection)url.openConnection();   
  3.   
  4. // 设置 User-Agent   
  5. httpConnection.setRequestProperty("User-Agent","NetFox");   
  6. // 设置断点续传的开始位置   
  7. httpConnection.setRequestProperty("RANGE","bytes=2000070");   
  8. // 获得输入流   
  9. InputStream input = httpConnection.getInputStream();  
URL url = new URL("http://www.sjtu.edu.cn/down.zip");
HttpURLConnection httpConnection = (HttpURLConnection)url.openConnection();

// 设置 User-Agent
httpConnection.setRequestProperty("User-Agent","NetFox");
// 设置断点续传的开始位置
httpConnection.setRequestProperty("RANGE","bytes=2000070");
// 获得输入流
InputStream input = httpConnection.getInputStream();


从输入流中取出的字节流就是 down.zip 文件从 2000070 开始的字节流。大家看,其实断点续传用 Java 实现起来还是很简单的吧。接下来要做的事就是怎么保存获得的流到文件中去了。

(2) 保存文件采用的方法。

我采用的是 IO 包中的 RandAccessFile 类。

操作相当简单,假设从 2000070 处开始保存文件,代码如下:




Java代码 ​HTTP断点续传_服务器​ ​HTTP断点续传_i++_02HTTP断点续传_服务器_03


  1. RandomAccess oSavedFile = new RandomAccessFile("down.zip","rw");   
  2. long nPos = 2000070;   
  3. // 定位文件指针到 nPos 位置   
  4. oSavedFile.seek(nPos);   
  5. byte[] b = new byte[1024];   
  6. int nRead;   
  7. // 从输入流中读入字节流,然后写到文件中   
  8. while((nRead=input.read(b,0,1024)) > 0)   
  9. {   
  10. oSavedFile.write(b,0,nRead);   
  11. }   
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 个类,包括一个测试类。

SiteFileFetch.java 负责整个文件的抓取,控制内部线程 (FileSplitterFetch 类 )。

FileSplitterFetch.java 负责部分文件的抓取。

FileAccess.java 负责文件的存储。

SiteInfoBean.java 要抓取的文件的信息,如文件保存的目录,名字,抓取文件的 URL 等。

Utility.java 工具类,放一些简单的方法。

TestMethod.java 测试类。

 

下面是源程序:




Java代码 ​HTTP断点续传_服务器​ ​HTTP断点续传_i++_02HTTP断点续传_服务器_03


  1. /**  
  2.  * SiteFileFetch.java   
  3.  */    
  4.  package NetFox;    
  5.  import java.io.*;    
  6.  import java.net.*;    
  7.  public class SiteFileFetch extends Thread {    
  8.      SiteInfoBean siteInfoBean = null; // 文件信息 Bean    
  9.      long[] nStartPos; // 开始位置   
  10.      long[] nEndPos; // 结束位置   
  11.      FileSplitterFetch[] fileSplitterFetch; // 子线程对象   
  12.      long nFileLength; // 文件长度   
  13.      boolean bFirst = true; // 是否第一次取文件   
  14.      boolean bStop = false; // 停止标志   
  15.      File tmpFile; // 文件下载的临时信息   
  16.      DataOutputStream output; // 输出到文件的输出流   
  17.      public SiteFileFetch(SiteInfoBean bean) throws IOException    
  18.      {    
  19.          siteInfoBean = bean;    
  20.          //tmpFile = File.createTempFile ("zhong","1111",new File(bean.getSFilePath()));    
  21.          tmpFile = new File(bean.getSFilePath()+File.separator + bean.getSFileName()+".info");   
  22.          if(tmpFile.exists ())    
  23.          {    
  24.              bFirst = false;    
  25.              read_nPos();    
  26.          }    
  27.          else    
  28.          {    
  29.              nStartPos = new long[bean.getNSplitter()];    
  30.              nEndPos = new long[bean.getNSplitter()];    
  31.          }    
  32.      }    
  33.      public void run()    
  34.      {    
  35.          // 获得文件长度   
  36.          // 分割文件   
  37.          // 实例 FileSplitterFetch    
  38.          // 启动 FileSplitterFetch 线程   
  39.          // 等待子线程返回   
  40.          try{    
  41.              if(bFirst)    
  42.              {    
  43.                  nFileLength = getFileSize();    
  44.                  if(nFileLength == -1)    
  45.                  {    
  46.                      System.err.println("File Length is not known!");    
  47.                  }    
  48.                  else if(nFileLength == -2)    
  49.                  {    
  50.                      System.err.println("File is not access!");    
  51.                  }    
  52.                  else    
  53.                  {    
  54.                  for(int i=0;i<nStartPos.length;i++)    
  55.                  {    
  56.                       nStartPos[i] = (long)(i*(nFileLength/nStartPos.length));    
  57.                  }    
  58.                 for(int i=0;i<nEndPos.length-1;i++)    
  59.                 {    
  60.                     nEndPos[i] = nStartPos[i+1];    
  61.                 }    
  62.                 nEndPos[nEndPos.length-1] = nFileLength;    
  63.             }    
  64.      }    
  65.  // 启动子线程   
  66.  fileSplitterFetch = new FileSplitterFetch[nStartPos.length];    
  67.  for(int i=0;i<nStartPos.length;i++)    
  68.  {    
  69.  fileSplitterFetch[i] = new FileSplitterFetch(siteInfoBean.getSSiteURL(),    
  70.  siteInfoBean.getSFilePath() + File.separator + siteInfoBean.getSFileName(),    
  71.  nStartPos[i],nEndPos[i],i);    
  72.  Utility.log("Thread " + i + " , nStartPos = " + nStartPos[i] + ", nEndPos = "    
  73.  + nEndPos[i]);    
  74.  fileSplitterFetch[i].start();    
  75.  }    
  76.  // fileSplitterFetch[nPos.length-1] = new FileSplitterFetch(siteInfoBean.getSSiteURL(),   
  77.  siteInfoBean.getSFilePath() + File.separator    
  78.  + siteInfoBean.getSFileName(),nPos[nPos.length-1],nFileLength,nPos.length-1);    
  79.  // Utility.log("Thread " +(nPos.length-1) + ",nStartPos = "+nPos[nPos.length-1]+",   
  80.  nEndPos = " + nFileLength);    
  81.  // fileSplitterFetch[nPos.length-1].start();    
  82.  // 等待子线程结束   
  83.  //int count = 0;    
  84.  // 是否结束 while 循环   
  85.  boolean breakWhile = false;    
  86.  while(!bStop)    
  87.  {    
  88.  write_nPos();    
  89.  Utility.sleep(500);    
  90.  breakWhile = true;    
  91.  for(int i=0;i<nStartPos.length;i++)    
  92.  {    
  93.  if(!fileSplitterFetch[i].bDownOver)    
  94.  {    
  95.  breakWhile = false;    
  96.  break;    
  97.  }    
  98.  }    
  99.  if(breakWhile)    
  100.  break;    
  101.  //count++;    
  102.  //if(count>4)    
  103.  // siteStop();    
  104.  }    
  105.  System.err.println("文件下载结束!");    
  106.  }    
  107.  catch(Exception e){e.printStackTrace ();}    
  108.  }    
  109.  // 获得文件长度   
  110.  public long getFileSize()    
  111.  {    
  112.  int nFileLength = -1;    
  113.  try{    
  114.  URL url = new URL(siteInfoBean.getSSiteURL());    
  115.  HttpURLConnection httpConnection = (HttpURLConnection)url.openConnection ();   
  116.  httpConnection.setRequestProperty("User-Agent","NetFox");    
  117.  int responseCode=httpConnection.getResponseCode();    
  118.  if(responseCode>=400)    
  119.  {    
  120.  processErrorCode(responseCode);    
  121.  return -2; //-2 represent access is error    
  122.  }    
  123.  String sHeader;    
  124.  for(int i=1;;i++)    
  125.  {    
  126.  //DataInputStream in = new DataInputStream(httpConnection.getInputStream ());    
  127.  //Utility.log(in.readLine());    
  128.  sHeader=httpConnection.getHeaderFieldKey(i);    
  129.  if(sHeader!=null)    
  130.  {    
  131.  if(sHeader.equals("Content-Length"))    
  132.  {    
  133.  nFileLength = Integer.parseInt(httpConnection.getHeaderField(sHeader));    
  134.  break;    
  135.  }    
  136.  }    
  137.  else    
  138.  break;    
  139.  }    
  140.  }    
  141.  catch(IOException e){e.printStackTrace ();}    
  142.  catch(Exception e){e.printStackTrace ();}    
  143.  Utility.log(nFileLength);    
  144.  return nFileLength;    
  145.  }    
  146.  // 保存下载信息(文件指针位置)   
  147.  private void write_nPos()    
  148.  {    
  149.  try{    
  150.  output = new DataOutputStream(new FileOutputStream(tmpFile));    
  151.  output.writeInt(nStartPos.length);    
  152.  for(int i=0;i<nStartPos.length;i++)    
  153.  {    
  154.  // output.writeLong(nPos[i]);    
  155.  output.writeLong(fileSplitterFetch[i].nStartPos);    
  156.  output.writeLong(fileSplitterFetch[i].nEndPos);    
  157.  }    
  158.  output.close();    
  159.  }    
  160.  catch(IOException e){e.printStackTrace ();}    
  161.  catch(Exception e){e.printStackTrace ();}    
  162.  }    
  163.  // 读取保存的下载信息(文件指针位置)   
  164.  private void read_nPos()    
  165.  {    
  166.  try{    
  167.  DataInputStream input = new DataInputStream(new FileInputStream(tmpFile));    
  168.  int nCount = input.readInt();    
  169.  nStartPos = new long[nCount];    
  170.  nEndPos = new long[nCount];    
  171.  for(int i=0;i<nStartPos.length;i++)    
  172.  {    
  173.  nStartPos[i] = input.readLong();    
  174.  nEndPos[i] = input.readLong();    
  175.  }    
  176.  input.close();    
  177.  }    
  178.  catch(IOException e){e.printStackTrace ();}    
  179.  catch(Exception e){e.printStackTrace ();}    
  180.  }    
  181.  private void processErrorCode(int nErrorCode)    
  182.  {    
  183.  System.err.println("Error Code : " + nErrorCode);    
  184.  }    
  185.  // 停止文件下载   
  186.  public void siteStop()    
  187.  {    
  188.  bStop = true;    
  189.  for(int i=0;i<nStartPos.length;i++)    
  190.  fileSplitterFetch[i].splitterStop();    
  191.  }    
  192.  }   
/**
* SiteFileFetch.java
*/
package NetFox;
import java.io.*;
import java.net.*;
public class SiteFileFetch extends Thread {
SiteInfoBean siteInfoBean = null; // 文件信息 Bean
long[] nStartPos; // 开始位置
long[] nEndPos; // 结束位置
FileSplitterFetch[] fileSplitterFetch; // 子线程对象
long nFileLength; // 文件长度
boolean bFirst = true; // 是否第一次取文件
boolean bStop = false; // 停止标志
File tmpFile; // 文件下载的临时信息
DataOutputStream output; // 输出到文件的输出流
public SiteFileFetch(SiteInfoBean bean) throws IOException
{
siteInfoBean = bean;
//tmpFile = File.createTempFile ("zhong","1111",new File(bean.getSFilePath()));
tmpFile = new File(bean.getSFilePath()+File.separator + bean.getSFileName()+".info");
if(tmpFile.exists ())
{
bFirst = false;
read_nPos();
}
else
{
nStartPos = new long[bean.getNSplitter()];
nEndPos = new long[bean.getNSplitter()];
}
}
public void run()
{
// 获得文件长度
// 分割文件
// 实例 FileSplitterFetch
// 启动 FileSplitterFetch 线程
// 等待子线程返回
try{
if(bFirst)
{
nFileLength = getFileSize();
if(nFileLength == -1)
{
System.err.println("File Length is not known!");
}
else if(nFileLength == -2)
{
System.err.println("File is not access!");
}
else
{
for(int i=0;i<nStartPos.length;i++)
{
nStartPos[i] = (long)(i*(nFileLength/nStartPos.length));
}
for(int i=0;i<nEndPos.length-1;i++)
{
nEndPos[i] = nStartPos[i+1];
}
nEndPos[nEndPos.length-1] = nFileLength;
}
}
// 启动子线程
fileSplitterFetch = new FileSplitterFetch[nStartPos.length];
for(int i=0;i<nStartPos.length;i++)
{
fileSplitterFetch[i] = new FileSplitterFetch(siteInfoBean.getSSiteURL(),
siteInfoBean.getSFilePath() + File.separator + siteInfoBean.getSFileName(),
nStartPos[i],nEndPos[i],i);
Utility.log("Thread " + i + " , nStartPos = " + nStartPos[i] + ", nEndPos = "
+ nEndPos[i]);
fileSplitterFetch[i].start();
}
// fileSplitterFetch[nPos.length-1] = new FileSplitterFetch(siteInfoBean.getSSiteURL(),
siteInfoBean.getSFilePath() + File.separator
+ siteInfoBean.getSFileName(),nPos[nPos.length-1],nFileLength,nPos.length-1);
// Utility.log("Thread " +(nPos.length-1) + ",nStartPos = "+nPos[nPos.length-1]+",
nEndPos = " + nFileLength);
// fileSplitterFetch[nPos.length-1].start();
// 等待子线程结束
//int count = 0;
// 是否结束 while 循环
boolean breakWhile = false;
while(!bStop)
{
write_nPos();
Utility.sleep(500);
breakWhile = true;
for(int i=0;i<nStartPos.length;i++)
{
if(!fileSplitterFetch[i].bDownOver)
{
breakWhile = false;
break;
}
}
if(breakWhile)
break;
//count++;
//if(count>4)
// siteStop();
}
System.err.println("文件下载结束!");
}
catch(Exception e){e.printStackTrace ();}
}
// 获得文件长度
public long getFileSize()
{
int nFileLength = -1;
try{
URL url = new URL(siteInfoBean.getSSiteURL());
HttpURLConnection httpConnection = (HttpURLConnection)url.openConnection ();
httpConnection.setRequestProperty("User-Agent","NetFox");
int responseCode=httpConnection.getResponseCode();
if(responseCode>=400)
{
processErrorCode(responseCode);
return -2; //-2 represent access is error
}
String sHeader;
for(int i=1;;i++)
{
//DataInputStream in = new DataInputStream(httpConnection.getInputStream ());
//Utility.log(in.readLine());
sHeader=httpConnection.getHeaderFieldKey(i);
if(sHeader!=null)
{
if(sHeader.equals("Content-Length"))
{
nFileLength = Integer.parseInt(httpConnection.getHeaderField(sHeader));
break;
}
}
else
break;
}
}
catch(IOException e){e.printStackTrace ();}
catch(Exception e){e.printStackTrace ();}
Utility.log(nFileLength);
return nFileLength;
}
// 保存下载信息(文件指针位置)
private void write_nPos()
{
try{
output = new DataOutputStream(new FileOutputStream(tmpFile));
output.writeInt(nStartPos.length);
for(int i=0;i<nStartPos.length;i++)
{
// output.writeLong(nPos[i]);
output.writeLong(fileSplitterFetch[i].nStartPos);
output.writeLong(fileSplitterFetch[i].nEndPos);
}
output.close();
}
catch(IOException e){e.printStackTrace ();}
catch(Exception e){e.printStackTrace ();}
}
// 读取保存的下载信息(文件指针位置)
private void read_nPos()
{
try{
DataInputStream input = new DataInputStream(new FileInputStream(tmpFile));
int nCount = input.readInt();
nStartPos = new long[nCount];
nEndPos = new long[nCount];
for(int i=0;i<nStartPos.length;i++)
{
nStartPos[i] = input.readLong();
nEndPos[i] = input.readLong();
}
input.close();
}
catch(IOException e){e.printStackTrace ();}
catch(Exception e){e.printStackTrace ();}
}
private void processErrorCode(int nErrorCode)
{
System.err.println("Error Code : " + nErrorCode);
}
// 停止文件下载
public void siteStop()
{
bStop = true;
for(int i=0;i<nStartPos.length;i++)
fileSplitterFetch[i].splitterStop();
}
}

 



Java代码 ​HTTP断点续传_服务器​ ​HTTP断点续传_i++_02HTTP断点续传_服务器_03


  1. /*   
  2.  **FileSplitterFetch.java   
  3.  */    
  4.  package NetFox;    
  5.  import java.io.*;    
  6.  import java.net.*;    
  7.  public class FileSplitterFetch extends Thread {    
  8.  String sURL; //File URL    
  9.  long nStartPos; //File Snippet Start Position    
  10.  long nEndPos; //File Snippet End Position    
  11.  int nThreadID; //Thread's ID    
  12.  boolean bDownOver = false; //Downing is over    
  13.  boolean bStop = false; //Stop identical    
  14.  FileAccessI fileAccessI = null; //File Access interface    
  15.  public FileSplitterFetch(String sURL,String sName,long nStart,long nEnd,int id)   
  16.  throws IOException    
  17.  {    
  18.  this.sURL = sURL;    
  19.  this.nStartPos = nStart;    
  20.  this.nEndPos = nEnd;    
  21.  nThreadID = id;    
  22.  fileAccessI = new FileAccessI(sName,nStartPos);    
  23.  }    
  24.  public void run()    
  25.  {    
  26.  while(nStartPos < nEndPos && !bStop)    
  27.  {    
  28.  try{    
  29.  URL url = new URL(sURL);    
  30.  HttpURLConnection httpConnection = (HttpURLConnection)url.openConnection ();    
  31.  httpConnection.setRequestProperty("User-Agent","NetFox");    
  32.  String sProperty = "bytes="+nStartPos+"-";    
  33.  httpConnection.setRequestProperty("RANGE",sProperty);    
  34.  Utility.log(sProperty);    
  35.  InputStream input = httpConnection.getInputStream();    
  36.  //logResponseHead(httpConnection);    
  37.  byte[] b = new byte[1024];    
  38.  int nRead;    
  39.  while((nRead=input.read(b,0,1024)) > 0 && nStartPos < nEndPos    
  40.  && !bStop)    
  41.  {    
  42.  nStartPos += fileAccessI.write(b,0,nRead);    
  43.  //if(nThreadID == 1)    
  44.  // Utility.log("nStartPos = " + nStartPos + ", nEndPos = " + nEndPos);    
  45.  }    
  46.  Utility.log("Thread " + nThreadID + " is over!");    
  47.  bDownOver = true;    
  48.  //nPos = fileAccessI.write (b,0,nRead);    
  49.  }    
  50.  catch(Exception e){e.printStackTrace ();}    
  51.  }    
  52.  }    
  53.  // 打印回应的头信息   
  54.  public void logResponseHead(HttpURLConnection con)    
  55.  {    
  56.  for(int i=1;;i++)    
  57.  {    
  58.  String header=con.getHeaderFieldKey(i);    
  59.  if(header!=null)    
  60.  //responseHeaders.put(header,httpConnection.getHeaderField(header));    
  61.  Utility.log(header+" : "+con.getHeaderField(header));    
  62.  else    
  63.  break;    
  64.  }    
  65.  }    
  66.  public void splitterStop()    
  67.  {    
  68.  bStop = true;    
  69.  }    
  70.  }    
  71.     
  72.  /*   
  73.  **FileAccess.java   
  74.  */    
  75.  package NetFox;    
  76.  import java.io.*;    
  77.  public class FileAccessI implements Serializable{    
  78.  RandomAccessFile oSavedFile;    
  79.  long nPos;    
  80.  public FileAccessI() throws IOException    
  81.  {    
  82.  this("",0);    
  83.  }    
  84.  public FileAccessI(String sName,long nPos) throws IOException    
  85.  {    
  86.  oSavedFile = new RandomAccessFile(sName,"rw");    
  87.  this.nPos = nPos;    
  88.  oSavedFile.seek(nPos);    
  89.  }    
  90.  public synchronized int write(byte[] b,int nStart,int nLen)    
  91.  {    
  92.  int n = -1;    
  93.  try{    
  94.  oSavedFile.write(b,nStart,nLen);    
  95.  n = nLen;    
  96.  }    
  97.  catch(IOException e)    
  98.  {    
  99.  e.printStackTrace ();    
  100.  }    
  101.  return n;    
  102.  }    
  103.  }    
  104.     
  105.  /*   
  106.  **SiteInfoBean.java   
  107.  */    
  108.  package NetFox;    
  109.  public class SiteInfoBean {    
  110.  private String sSiteURL; //Site's URL    
  111.  private String sFilePath; //Saved File's Path    
  112.  private String sFileName; //Saved File's Name    
  113.  private int nSplitter; //Count of Splited Downloading File    
  114.  public SiteInfoBean()    
  115.  {    
  116.  //default value of nSplitter is 5    
  117.  this("","","",5);    
  118.  }    
  119.  public SiteInfoBean(String sURL,String sPath,String sName,int nSpiltter)   
  120.  {    
  121.  sSiteURL= sURL;    
  122.  sFilePath = sPath;    
  123.  sFileName = sName;    
  124.  this.nSplitter = nSpiltter;    
  125.  }    
  126.  public String getSSiteURL()    
  127.  {    
  128.  return sSiteURL;    
  129.  }    
  130.  public void setSSiteURL(String value)    
  131.  {    
  132.  sSiteURL = value;    
  133.  }    
  134.  public String getSFilePath()    
  135.  {    
  136.  return sFilePath;    
  137.  }    
  138.  public void setSFilePath(String value)    
  139.  {    
  140.  sFilePath = value;    
  141.  }    
  142.  public String getSFileName()    
  143.  {    
  144.  return sFileName;    
  145.  }    
  146.  public void setSFileName(String value)    
  147.  {    
  148.  sFileName = value;    
  149.  }    
  150.  public int getNSplitter()    
  151.  {    
  152.  return nSplitter;    
  153.  }    
  154.  public void setNSplitter(int nCount)    
  155.  {    
  156.  nSplitter = nCount;    
  157.  }    
  158.  }    
  159.     
  160.  /*   
  161.  **Utility.java   
  162.  */    
  163.  package NetFox;    
  164.  public class Utility {    
  165.  public Utility()    
  166.  {    
  167.  }    
  168.  public static void sleep(int nSecond)    
  169.  {    
  170.  try{    
  171.  Thread.sleep(nSecond);    
  172.  }    
  173.  catch(Exception e)    
  174.  {    
  175.  e.printStackTrace ();    
  176.  }    
  177.  }    
  178.  public static void log(String sMsg)    
  179.  {    
  180.  System.err.println(sMsg);    
  181.  }    
  182.  public static void log(int sMsg)    
  183.  {    
  184.  System.err.println(sMsg);    
  185.  }    
  186.  }    
  187.     
  188.  /*   
  189.  **TestMethod.java   
  190.  */    
  191.  package NetFox;    
  192.  public class TestMethod {    
  193.  public TestMethod()    
  194.  { ///xx/weblogic60b2_win.exe    
  195.  try{    
  196.  SiteInfoBean bean = new SiteInfoBean("http://localhost/xx/weblogic60b2_win.exe",   
  197.      "L:\\temp","weblogic60b2_win.exe",5);    
  198.  //SiteInfoBean bean = new SiteInfoBean("http://localhost:8080/down.zip","L:\\temp",   
  199.      "weblogic60b2_win.exe",5);    
  200.  SiteFileFetch fileFetch = new SiteFileFetch(bean);    
  201.  fileFetch.start();    
  202.  }    
  203.  catch(Exception e){e.printStackTrace ();}    
  204.  }    
  205.  public static void main(String[] args)    
  206.  {    
  207.  new TestMethod();    
  208.  }    
  209.  }  
/* 
**FileSplitterFetch.java
*/
package NetFox;
import java.io.*;
import java.net.*;
public class FileSplitterFetch extends Thread {
String sURL; //File URL
long nStartPos; //File Snippet Start Position
long nEndPos; //File Snippet End Position
int nThreadID; //Thread's ID
boolean bDownOver = false; //Downing is over
boolean bStop = false; //Stop identical
FileAccessI fileAccessI = null; //File Access interface
public FileSplitterFetch(String sURL,String sName,long nStart,long nEnd,int id)
throws IOException
{
this.sURL = sURL;
this.nStartPos = nStart;
this.nEndPos = nEnd;
nThreadID = id;
fileAccessI = new FileAccessI(sName,nStartPos);
}
public void run()
{
while(nStartPos < nEndPos && !bStop)
{
try{
URL url = new URL(sURL);
HttpURLConnection httpConnection = (HttpURLConnection)url.openConnection ();
httpConnection.setRequestProperty("User-Agent","NetFox");
String sProperty = "bytes="+nStartPos+"-";
httpConnection.setRequestProperty("RANGE",sProperty);
Utility.log(sProperty);
InputStream input = httpConnection.getInputStream();
//logResponseHead(httpConnection);
byte[] b = new byte[1024];
int nRead;
while((nRead=input.read(b,0,1024)) > 0 && nStartPos < nEndPos
&& !bStop)
{
nStartPos += fileAccessI.write(b,0,nRead);
//if(nThreadID == 1)
// Utility.log("nStartPos = " + nStartPos + ", nEndPos = " + nEndPos);
}
Utility.log("Thread " + nThreadID + " is over!");
bDownOver = true;
//nPos = fileAccessI.write (b,0,nRead);
}
catch(Exception e){e.printStackTrace ();}
}
}
// 打印回应的头信息
public void logResponseHead(HttpURLConnection con)
{
for(int i=1;;i++)
{
String header=con.getHeaderFieldKey(i);
if(header!=null)
//responseHeaders.put(header,httpConnection.getHeaderField(header));
Utility.log(header+" : "+con.getHeaderField(header));
else
break;
}
}
public void splitterStop()
{
bStop = true;
}
}

/*
**FileAccess.java
*/
package NetFox;
import java.io.*;
public class FileAccessI implements Serializable{
RandomAccessFile oSavedFile;
long nPos;
public FileAccessI() throws IOException
{
this("",0);
}
public FileAccessI(String sName,long nPos) throws IOException
{
oSavedFile = new RandomAccessFile(sName,"rw");
this.nPos = nPos;
oSavedFile.seek(nPos);
}
public synchronized int write(byte[] b,int nStart,int nLen)
{
int n = -1;
try{
oSavedFile.write(b,nStart,nLen);
n = nLen;
}
catch(IOException e)
{
e.printStackTrace ();
}
return n;
}
}

/*
**SiteInfoBean.java
*/
package NetFox;
public class SiteInfoBean {
private String sSiteURL; //Site's URL
private String sFilePath; //Saved File's Path
private String sFileName; //Saved File's Name
private int nSplitter; //Count of Splited Downloading File
public SiteInfoBean()
{
//default value of nSplitter is 5
this("","","",5);
}
public SiteInfoBean(String sURL,String sPath,String sName,int nSpiltter)
{
sSiteURL= sURL;
sFilePath = sPath;
sFileName = sName;
this.nSplitter = nSpiltter;
}
public String getSSiteURL()
{
return sSiteURL;
}
public void setSSiteURL(String value)
{
sSiteURL = value;
}
public String getSFilePath()
{
return sFilePath;
}
public void setSFilePath(String value)
{
sFilePath = value;
}
public String getSFileName()
{
return sFileName;
}
public void setSFileName(String value)
{
sFileName = value;
}
public int getNSplitter()
{
return nSplitter;
}
public void setNSplitter(int nCount)
{
nSplitter = nCount;
}
}

/*
**Utility.java
*/
package NetFox;
public class Utility {
public Utility()
{
}
public static void sleep(int nSecond)
{
try{
Thread.sleep(nSecond);
}
catch(Exception e)
{
e.printStackTrace ();
}
}
public static void log(String sMsg)
{
System.err.println(sMsg);
}
public static void log(int sMsg)
{
System.err.println(sMsg);
}
}

/*
**TestMethod.java
*/
package NetFox;
public class TestMethod {
public TestMethod()
{ ///xx/weblogic60b2_win.exe
try{
SiteInfoBean bean = new SiteInfoBean("http://localhost/xx/weblogic60b2_win.exe",
"L:\\temp","weblogic60b2_win.exe",5);
//SiteInfoBean bean = new SiteInfoBean("http://localhost:8080/down.zip","L:\\temp",
"weblogic60b2_win.exe",5);
SiteFileFetch fileFetch = new SiteFileFetch(bean);
fileFetch.start();
}
catch(Exception e){e.printStackTrace ();}
}
public static void main(String[] args)
{
new TestMethod();
}
}