1. 为什么需要多线程下载?

    服务器的资源有限,同时的平均地分配给每个客户端。开启的线程越多抢占的服务的资源就越多,下载的速度就越块。

2. 下载速度的限制条件?

(1)你的电脑手机宽带的带宽。(网络运营商给用户的限制)

(2)服务器上传的带宽限制。 (服务器端资源获取速度的限制)----迅雷, p2p快播等下载,可以同时间使用多台服务器帮助用户下载资源,速度自然会加快。

注意:并不是开的线程越多下载速度越快,可能会消耗大量时间在线程调度上。

Android下推荐开启: ~ 线程。

3. 如何进行多线程的下载:

(1)在客户端本地创建一个空文件(申请一块内存),大小要和服务器上要下载的资源一样.

(2)开启3个线程,都去下载服务器的数据.

(3)当三个线程都工作完毕后,多线程的下载就结束了.

这里特别注意最后一个线程需要修正,主要是因为不可能实现完全等分,具体如下:

4.JavaSE代码实现多线程下载:

 (1)我们可以先编写java项目工程,调试实现多线程下载逻辑类MutilDownloader.java:

(2)打开Apache服务器,在相应的目录下存放测试下载文件,如下:

(3)MutilDownloader.java,如下:

 package com.himi.mutildownload;

 import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL; /**
* 多线程的下载器
*
*/
public class MutilDownloader {
/**
* Apache服务器上资源下载的路径
*/
private static final String path = "http://49.123.76.170/movies/test.avi";
/**
* 多少个线程去下载服务器的资源
*/
private static int threadCount = 4; /**
* 正在运行的线程的数量
*/
private static int runningThreadCount; public static void main(String[] args) throws Exception {
URL url = new URL(path);
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
conn.setRequestMethod("GET");
int code = conn.getResponseCode();
if (code == 200) {
int length = conn.getContentLength();
System.out.println("服务器文件的大小为:" + length); // 1. 创建一个空白文件文件的大小和服务器资源一样
RandomAccessFile raf = new RandomAccessFile(getFileName(path), "rw");
raf.setLength(length);
raf.close(); // 每个线程下载的平均区块大小
int blocksize = length / threadCount;
System.out.println("每一份:" + blocksize); runningThreadCount = threadCount;
// 2. 开启3个线程,都去下载服务器的对应数据
for (int threadId = 0; threadId < threadCount; threadId++) {
int startIndex = threadId * blocksize;
int endIndex = (threadId + 1) * blocksize - 1; // 最后一个线程的修正,最后一个线程endIndex设置为文件末尾
if (threadId == (threadCount - 1)) {
endIndex = length - 1;// 文件byte是从0开始计数的
} new DownloadThread(startIndex, endIndex, threadId).start();
}
} // 3. 当三个线程都工作完毕后,多线程的下载就结束了. } public static class DownloadThread extends Thread {
/**
* 线程id
*/
int threadId; /**
* 当前线程下载的开始位置
*/
int startIndex;
/**
* 当前线程下载的结束位置
*/
int endIndex; /**
* 当前线程下载到文件的位置
*/
int filePosition; /**
*
* @param startIndex
* 开始位置
* @param endIndex
* 结束位置
* @param threadId
* 线程id
*/
public DownloadThread(int startIndex, int endIndex, int threadId) {
this.startIndex = startIndex;
this.endIndex = endIndex;
this.threadId = threadId;
filePosition = startIndex;
} @Override
public void run() {
try {
// 用一个文本记录当前线程下载的进程
File file = new File(threadId + getFileName(path) + ".txt"); if (file.exists() && file.length() > 0) {
FileInputStream fis = new FileInputStream(file);
BufferedReader br = new BufferedReader(new InputStreamReader(fis));
filePosition = Integer.parseInt(br.readLine());// 上一次下载到文件的哪个位子。
startIndex = filePosition;
fis.close();
} System.out.println("线程:" + threadId + "实际上下载的位置:" + startIndex + "~~~" + endIndex); URL url = new URL(path);
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
conn.setRequestMethod("GET");
// 指定从服务器下载的范围,http请求的头
conn.setRequestProperty("Range", "bytes=" + startIndex + "-" + endIndex); int code = conn.getResponseCode();// 2XX成功 3XX重定向 4XX资源找不到
// 5XX服务器异常
if (code == 206) {// 206:表示请求部分数据成功
// 返回服务器端对应数据的输入流
InputStream is = conn.getInputStream();
RandomAccessFile raf = new RandomAccessFile(getFileName(path), "rwd"); /**
* ☆☆☆☆☆非常重要☆☆☆☆☆ 一定要记得定位文件写的位置
* 不同线程在文件中(代码开头:创建的空文件)开始写入的位置是不一样的
*/
raf.seek(startIndex);
byte[] buffer = new byte[1024 * 1024 * 10];
int len = -1;
while ((len = is.read(buffer)) != -1) {
raf.write(buffer, 0, len);
filePosition += len;
RandomAccessFile rafinfo = new RandomAccessFile(file, "rwd");
rafinfo.write(String.valueOf(filePosition).getBytes());
rafinfo.close();
}
raf.close();
is.close(); System.out.println("线程:" + threadId + "下载完毕了。"); }
} catch (Exception e) {
e.printStackTrace();
} finally { // 三个线程都结束了,下载完毕
synchronized (MutilDownloader.class) {
runningThreadCount--;
if (runningThreadCount == 0) {
System.out.println("所有的线程都下载完毕了");
for (int i = 0; i < threadCount; i++) {
File f = new File(i + getFileName(path) + ".txt");
System.out.println(f.delete());
}
}
}
}
}
} /**
* 获取路径对应的文件名
*
* @param path
* @return
*/
private static String getFileName(String path) {
int beginIndex = path.lastIndexOf("/") + 1;
return path.substring(beginIndex);
} }

需要特别注意的是:RandomAccessFile.seek(startIndex),它是用来定位文件写入的位置。

运行程序,观察Console,如下:

刷新Java工程项目,如下:

双击打开test.avi,发现是可以播放的。

Android(java)学习笔记215:多线程断点下载的原理(JavaSE实现)的更多相关文章

  1. 我的Android进阶之旅------>Android基于HTTP协议的多线程断点下载器的实现

    一.首先写这篇文章之前,要了解实现该Android多线程断点下载器的几个知识点 1.多线程下载的原理,如下图所示 注意:由于Android移动设备和PC机的处理器还是不能相比,所以开辟的子线程建议不要 ...

  2. Android(java)学习笔记216:多线程断点下载的原理(Android实现)

    之前在Android(java)学习笔记215中,我们从JavaSE的角度去实现了多线程断点下载,下面从Android角度实现这个断点下载: 1.新建一个Android工程: (1)其中我们先实现布局 ...

  3. Android(java)学习笔记159:多线程断点下载的原理(Android实现)

    之前在Android(java)学习笔记215中,我们从JavaSE的角度去实现了多线程断点下载,下面从Android角度实现这个断点下载: 1. 新建一个Android工程: (1)其中我们先实现布 ...

  4. Android(java)学习笔记158:多线程断点下载的原理(JavaSE实现)

    1. 为什么需要多线程下载?     服务器的资源有限,同时的平均地分配给每个客户端.开启的线程越多抢占的服务的资源就越多,下载的速度就越块. 2. 下载速度的限制条件? (1)你的电脑手机宽带的带宽 ...

  5. 【原】Java学习笔记032 - 多线程

    package cn.temptation; public class Sample01 { public static void main(String[] args) { /* * [进程]:正在 ...

  6. Java学习笔记之——多线程

    多线程编程 程序: 进程:一个程序运行就会产生一个进程 线程:进程的执行流程,一个进程至少有一个线程,称为主线程 如:QQ聊着天,同时在听音乐 一个进程可以有多个线程,多个线程共享同一个进程的资源 线 ...

  7. Java学习笔记:多线程(一)

    Java中线程的五种状态: 新建状态(New) 就绪状态(Runnable) 运行状态(Running) 阻塞状态(Blocked) 凋亡状态(Dead) 其中阻塞状态(Blocked)又分为三种: ...

  8. java学习笔记(5)多线程

    一.简介(过段时间再写,多线程难度有点大) --------------------------------------- 1.进程:运行时的概念,运行的应用程序 2.线程:应用程序内部并发执行的代码 ...

  9. Java 学习笔记(11)——多线程

    Java内部提供了针对多线程的支持,线程是CPU执行的最小单位,在多核CPU中使用多线程,能够做到多个任务并行执行,提高效率. 使用多线程的方法 创建Thread类的子类,并重写run方法,在需要启动 ...

随机推荐

  1. SQL 左外连接查询 将右表中的多行变为左表的一列或多列

    示例: --行列互转 /**************************************************************************************** ...

  2. QTP10破解

    1. 下载QTP10.0版本 http://h30302.www3.hp.com/prdownloads/T6510-15063.zip?ordernumber=380454070&itemi ...

  3. 面试小结(java基础)

    一个.java源文件中可以有多个类吗?(内部类除外)有什么条件?带着这个疑惑,动手建几个测试类, 揭开心中的疑惑.以下是解开疑惑过程: package test;/** * 一个.java源文件中可以 ...

  4. go程序性能优化

    性能优化总结: 1 尽量避免频繁创建对象,即减少&{},new,make的使用2 数组可当切片用,当需要使用切片时,可考虑能使用数组来减少切片的创建3 当某类临时对象被多个协频繁程使用时,可用 ...

  5. Spring 配置自动扫描spring bean配置

    <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w ...

  6. 一篇故事讲述了计算机网络里的基本概念:网关,DHCP,IP寻址,ARP欺骗,路由,DDOS等...

    计算机主机网关的作用是什么? 假设你的名字叫小不点,你住在一个大院子里,你的邻居有很多小伙伴,在门口传达室还有个看大门的李大爷,李大爷就是你的网关.当你想跟院子里的某个小伙伴玩,只要你在院子里大喊一声 ...

  7. Hibernate 的*.hbm.xml文件的填写技巧

    ================================================================================= 模板: <!-- ?属性,本类 ...

  8. J2EE和EJB有什么关系?

    其实j2EE就是java的企业版,与javaSE(标准版)有是有区别的,所以运行环境会有差异(简单点可以说,这是两组不同的接口),我们一般所熟悉的tomcat仅仅只实现了j2ee的一小部分规范,它只是 ...

  9. 14.5.3 Locks Set by Different SQL Statements in InnoDB

    14.5.3 Locks Set by Different SQL Statements in InnoDB 通过不同的SQL语句设置的锁 在InnoDB中 一个锁定读, 一个UPDATE 或者一个D ...

  10. libc.so.6 误删后修复

    libc.so.6 误删后修复  libc.so.6 被删除了(libc.so.6只是个链接,真实的lib 文件是 libc-2.15.so) su, sudo,ls, cp, mv 等等一系列命令都 ...