该资源来源于李刚老师的疯狂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 block e.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、依次创建、启动多条线程来下载网路资源的指定部分。