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

URL 多线程下载的更多相关文章

  1. 多线程下载图片,同步下载http://www.importnew.com/15731.html

    package mutiDownload; import java.io.IOException; import java.io.InputStream; import java.io.RandomA ...

  2. Android学习记录(4)—在java中学习多线程下载的基本原理和基本用法①

    多线程下载在我们生活中非常常见,比如迅雷就是我们常用的多线程的下载工具,当然还有断点续传,断点续传我们在下一节来讲,android手机端下载文件时也可以用多线程下载,我们这里是在java中写一个测试, ...

  3. Java多线程下载文件

    package com.test.download;   import java.io.File; import java.io.InputStream; import java.io.RandomA ...

  4. java 网络编程基础 InetAddress类;URLDecoder和URLEncoder;URL和URLConnection;多线程下载文件示例

    什么是IPV4,什么是IPV6: IPv4使用32个二进制位在网络上创建单个唯一地址.IPv4地址由四个数字表示,用点分隔.每个数字都是十进制(以10为基底)表示的八位二进制(以2为基底)数字,例如: ...

  5. 【Java EE 学习 22 下】【单线程下载】【单线程断点下载】【多线程下载】

    一.文件下载简述 1.使用浏览器从网页上下载文件,Servlet需要增加一些响应头信息 (1)response.setContentType("application/force-downl ...

  6. Java--使用多线程下载,断点续传技术原理(RandomAccessFile)

    一.基础知识 1.什么是线程?什么是进程?它们之间的关系? 可以参考之前的一篇文章:java核心知识点学习----并发和并行的区别,进程和线程的区别,如何创建线程和线程的四种状态,什么是线程计时器 简 ...

  7. android 多线程下载 断点续传

    来源:网易云课堂Android极客班第八次作业练习 练习内容: 多线程 asyncTask handler 多线程下载的原理 首先获取到目标文件的大小,然后在磁盘上申请一块空间用于保存目标文件,接着把 ...

  8. 无废话Android之smartimageview使用、android多线程下载、显式意图激活另外一个activity,检查网络是否可用定位到网络的位置、隐式意图激活另外一个activity、隐式意图的配置,自定义隐式意图、在不同activity之间数据传递(5)

    1.smartimageview使用 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android&q ...

  9. Java 仿迅雷多线程下载

    package net.webjoy.jackluo.android_json; /** * 1.http Range "bytes="+ start+end * 2.Random ...

随机推荐

  1. Eclipse中配置javap命令

    Run→External Tools→External Tools Configurations-进入如下图二所示的Program配置界面.也可以通过如下图一所示的工具栏按钮进入Program配置界面 ...

  2. syslog(),closelog()与openlog()--日志操作函数

    在典型的 LINUX 安装中,/var/log/messages 包含所有的系统消息,/var/log/mail 包含来自邮件系统的其它日志消息,/var/log/debug 可能包含调试消息.根据你 ...

  3. VMware 下的CentOS6.7 虚拟机与Windows7通信

    在有网络的情况下,VMware 虚拟机使用桥接模式(Bridged) 和NAT方式,会自动通信,但是在没有网络的情况下怎么办呢?对,是的,使用host-only模式,如何设置呢? 注:将Windows ...

  4. 有奖试读—Windows PowerShell实战指南(第2版)

    为什么要学PowerShell? Windows用户都已习惯于使用图形化界面去完成工作,因为GUI总能轻易地实现很多功能,并且不需要记住很多命令.使得短时间学会一种工具成为可能. 但是不幸的是,GUI ...

  5. Python 自动刷博客浏览量

    哈哈,今天的话题有点那什么了哈.咱们应该秉承学习技术的角度来看,那么就开始今天的话题吧. 思路来源 今天很偶然的一个机会,听到别人在谈论现在的"刷量"行为,于是就激发了我的好奇心. ...

  6. Linux 内存管理之highmem简介

    一.Linux内核地址空间 一般来说Linux 内核按照 3:1 的比率来划分虚拟内存(X86等):3 GB 的虚拟内存用于用户空间,1GB 的内存用于内核空间.当然有些体系结构如MIPS使用2:2 ...

  7. SSH深度历险(九) Struts2+DWZ+Uploadify实现多文件(文件和图片等等)上传

    在gxpt_uas系统中,要实现文件(文件和图片等等,可以灵活配置)的批量上传至mongodb,在学习这个过程中,学习了mongodb,并实现了批量上传的功能,实现思路:在DWZ的基础上参考官方的实例 ...

  8. 任务定义器——SocketProcessor

    将socket扔进线程池前需要定义好任务,要进行哪些逻辑处理由SocketProcessor定义,根据线程池的约定,作为任务必须扩展Runnable.用如下伪代码表示 protected class ...

  9. linq---我为你提笔序,你的美不只查询语句

    LinQ百度百科对她这样解释,是一组用于c#和Visual Basic语言的扩展.它允许编写C#或者Visual Basic代码以查询数据库相同的方式操作内存数据. LINQ是Language Int ...

  10. 《java入门第一季》之Socket编程通信和TCP协议通信图解

    Socket编程通信图解原理: TCP协议通信图解