URL 多线程下载

来源:互联网 发布:淘宝给差评被报复 编辑:程序博客网 时间:2024/06/05 22:58

该资源来源于李刚老师的疯狂JAVA讲义

InutStream openStream():打开与此URL链接,并返回一个用于读取该URL资源的InputStream.

提供的openStream()可以读取该URL资源的InputStream,通过该 方法可以非常方便的读取远程资源--甚至实现多线程下载。程序如下:

package com.net;import java.io.IOException;import java.io.InputStream;import java.io.RandomAccessFile;import java.net.MalformedURLException;import java.net.URL;import java.net.URLConnection;/** * 多线程下载 * @author yulei * *///定义下载从start到end 的内容的线程class DownThread extends Thread{//定义字节数组(取水的竹筒)的长度private final int BUF_LEN=32;//定义下载的起始点private long start;//定义下载的结束点private long end;//下载资源对应的输入流private InputStream is;//将下载的字节输出到raf中private RandomAccessFile raf;//构造器,传入输入流,输出流和下载起始点、结束点public DownThread(long start,long end ,InputStream is ,RandomAccessFile raf){//输出改线程负责下载的字节位置System.out.println(start+"---->"+end);this.start=start;this.end=end;this.is=is;this.raf=raf;}public void run(){try {is.skip(start);raf.seek(start);//定义读取输入流内容的缓存数组byte[] buff=new byte[BUF_LEN];//本线程负责下载资源的大小long contentLen=end-start;//定义最多需要读取几次就可以完成本线程的下载long times=contentLen/BUF_LEN+4;//实际读取的字节数int hasRead=0;for (int i=0;i<times;i++){hasRead=is.read(buff);//如果读取的字节数小于0,则退出循环!if(hasRead<0){break;}raf.write(buff, 0, hasRead);}} catch (Exception e) {e.printStackTrace();}//使用finally块来关闭当前线程的输入流、输出流finally {try {  if(is!=null){  is.close();   }  if(raf!=null){  raf.close();  }} catch (Exception e2) {e2.printStackTrace();}}}}public class MutilDown { public static void main(String[] args) {  final int DOWN_TREAD_NUM=4; final String OUT_FILE_NAME="down.jpg"; InputStream[] isArr=new InputStream[DOWN_TREAD_NUM]; RandomAccessFile[] outArr=new RandomAccessFile[DOWN_TREAD_NUM]; try {  //创建一个URL对象  URL url=new URL("http://www.baidu.com/img/bd_logo1.png");  //以此URL对象打开一个输入流  isArr[0]=url.openStream();  long fileLen=getFileLength(url);  System.out.println("网路资源的大小:"+fileLen);  //以输出文件名创建第一个RandomAccessFile输出流  outArr[0]=new RandomAccessFile(OUT_FILE_NAME, "rw");  //创建一个与下载资源相同大小的空文件  for (int i=0;i<fileLen;i++){  outArr[0].write(0);  }  //每线程应该下载的字节数  long numPerThread=fileLen/DOWN_TREAD_NUM;  //整个下载资源整除后剩下的余数  long left=fileLen%DOWN_TREAD_NUM;  for(int i=0;i<DOWN_TREAD_NUM;i++){  //为每个线程打开一个输入流、一个RandomAccessFile对象  //让每个线程分别负责下载资源的不同部分  if(i!=0){ //上面已经初始化一次了,所以这里写i!=0 即可  //以URL打开多个输入流  isArr[i]=url.openStream();  //以指定输出文件创建多个RandomAccessFile对象  outArr[i]=new RandomAccessFile(OUT_FILE_NAME,"rw");  }    //分别启动多个线程下载网路资源  if(i==DOWN_TREAD_NUM-1){  //最后一个线程下载指定numPerThread+left个字节  new DownThread(i*numPerThread, (i+1)*numPerThread, isArr[i], outArr[i]).start();  }else{  //每个线程负责下载一定的numPerThread个字节  new DownThread(i*numPerThread,(i+1)*numPerThread,isArr[i], outArr[i]).start();  }    }  } catch (MalformedURLException e) {e.printStackTrace();} catch (IOException e) {// TODO Auto-generated catch blocke.printStackTrace();}}  //定义获取指定网路资源长度的方法  public static long getFileLength(URL url) throws IOException{ long lenth=0; //打开该URL对应的URLConnection URLConnection con=url.openConnection(); //获取连接URL资源的长度 long size =con.getContentLength(); System.out.println("网路资源SIZE 的长度:"+size); lenth=size;return lenth; } }

 

上面程序定义了DownTread线程类,该线程从InputStream中读取从start开始,到end结束的所有字节数据,并写入RandomAccessFile对象。这个DownThread线程类的run就是一个简单的输入、输出实现。

程序中MutilDown类中的main方法负责按如下步骤来实现多线程下载:

1 、创建URL对象。

2、获取指定URL对象所指向资源的大小(由getFIleLength方法实现),此处用到了URLConnection类,该类代表JAVA应用程序和URL之间的通信链接。

3、在本地磁盘上创建一个与网路资源相同大小的空文件。

4、计算每条线程应该下载网路资源的哪个部分(从哪个字节开始,到哪个字节结束)。

5、依次创建、启动多条线程来下载网路资源的指定部分。

 

 

 

 

 

 

 

0 0
原创粉丝点击