Android(Java):多线程断点续传下载

来源:互联网 发布:淘宝手机开店认证步骤 编辑:程序博客网 时间:2024/05/22 05:10

使用多线程下载文件可以更快完成文件的下载,多线程下载文件之所以快,是因为其抢占的服务器资源多。如:假设服务器同时最多服务100个用户,在服务器中一条线程对应一个用户,100条线程在计算机中并非并发执行,而是由CPU划分时间片轮流执行,如果A应用使用了99条线程下载文件,那么相当于占用了99个用户的资源,假设一秒内CPU分配给每条线程的平均执行时间是10ms,A应用在服务器中一秒内就得到了990ms的执行时间,而其他应用在一秒内只有10ms的执行时间。就如同一个水龙头,每秒出水量相等的情况下,放990毫秒的水

肯定比放10毫秒的水要多。

多线程下载的实现过程:

1>首先得到下载文件的长度,然后设置本地文件

的长度。

HttpURLConnection.getContentLength();

RandomAccessFile file = newRandomAccessFile("QQWubiSetup.exe","rwd");

file.setLength(filesize);//设置本地文件的长度

2>根据文件长度和线程数计算每条线程下载的数据长度和下载位置。如:文件的长度为6M,线程数为3,那么,每条线程下载的数据长度为2M,每条线程开始下载的位置如上图所示。

3>使用HttpRange头字段指定每条线程从文件的什么位置开始下载,下载到什么位置为止,如:指定从文件的2M位置开始下载,下载到位置(4M-1byte)为止,代码如下:

HttpURLConnection.setRequestProperty("Range", "bytes=2097152-4194303");

4>保存文件,使用RandomAccessFile类指定每条线程从本地文件的什么位置开始写入数据。

RandomAccessFilethreadfile = newRandomAccessFile("QQWubiSetup.exe ","rwd");

threadfile.seek(2097152);//从文件的什么位置开始写入数据

 

public class FileDownLoader {

  @Test

  public void download() throws Exception {

  String path = "http://net.itcast.cn/QQWubiSetup.exe";

  URL url = new URL(path);

  HttpURLConnection conn = (HttpURLConnection) url.openConnection();

  conn.setConnectTimeout(5*1000);

  conn.setRequestMethod("GET");

  conn.setRequestProperty("Accept", "image/gif, image/jpeg, image/pjpeg, image/pjpeg, application/x-shockwave-flash, application/xaml+xml, application/vnd.ms-xpsdocument, application/x-ms-xbap, application/x-ms-application, application/vnd.ms-excel, application/vnd.ms-powerpoint, application/msword, */*");

  conn.setRequestProperty("Accept-Language", "zh-CN");

  conn.setRequestProperty("Charset", "UTF-8");

  conn.setRequestProperty("User-Agent", "Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 5.2; Trident/4.0; .NET CLR 1.1.4322; .NET CLR 2.0.50727; .NET CLR 3.0.04506.30; .NET CLR 3.0.4506.2152; .NET CLR 3.5.30729)");

  conn.setRequestProperty("Connection", "Keep-Alive");

  System.out.println(conn.getResponseCode());

 

  int filesize = conn.getContentLength();//得到文件大小

  conn.disconnect();

  int threasize = 3;//线程数

  int perthreadsize = filesize / 3 + 1;

  RandomAccessFile file = new RandomAccessFile("102.wma","rw");

  file.setLength(filesize);//设置本地文件的大小

  file.close();

  for(int i=0; i<threasize ; i++){

  int startpos = i * perthreadsize;//计算每条线程的下载位置

  RandomAccessFile perthreadfile = new RandomAccessFile("102.wma","rw");//

  perthreadfile.seek(startpos);//从文件的什么位置开始写入数据

  new DownladerThread(i, path, startpos, perthreadsize, perthreadfile).start();

  }

  //以下代码要求用户输入q才会退出测试方法,如果没有下面代码,会因为进程结束而导致进程内的下载线程被销毁

  int quit = System.in.read();

  while('q'!=quit){

  Thread.sleep(2 * 1000);

  }

  }

 

  private class DownladerThread extends Thread{

  private int startpos;//从文件的什么位置开始下载

  private int perthreadsize;//每条线程需要下载的文件大小

  private String path;

  private RandomAccessFile file;

  private int threadid;

 

  public DownladerThread(int threadid, String path, int startpos, int perthreadsize, RandomAccessFile perthreadfile) {

  this.path = path;

  this.startpos = startpos;

  this.perthreadsize = perthreadsize;

  this.file = perthreadfile;

  this.threadid = threadid;

  }

 

  @Override

  public void run() {

  try {

  URL url = new URL(path);

  HttpURLConnection conn = (HttpURLConnection)url.openConnection();

  conn.setConnectTimeout(5 * 1000);

  conn.setRequestMethod("GET");

  conn.setRequestProperty("Accept", "image/gif, image/jpeg, image/pjpeg, image/pjpeg, application/x-shockwave-flash, application/xaml+xml, application/vnd.ms-xpsdocument, application/x-ms-xbap, application/x-ms-application, application/vnd.ms-excel, application/vnd.ms-powerpoint, application/msword, */*");

  conn.setRequestProperty("Accept-Language", "zh-CN");

  conn.setRequestProperty("Charset", "UTF-8");

  conn.setRequestProperty("Range", "bytes=" + this.startpos + "-");

  InputStream inStream = conn.getInputStream();

  byte[] buffer = new byte[1024];

  int len = 0;

  int length = 0;

  while(length<perthreadsize && (len = inStream.read(buffer))!=-1){

  file.write(buffer, 0, len);

  length += len;//累计该线程下载的总大小

  }

  file.close();

  inStream.close();

  System.out.println(threadid+ "线程完成下载");

  } catch (Exception e) {

  e.printStackTrace();

  }

  }

  }

}

 

原创粉丝点击