Android(java)学习笔记215:多线程断点下载的原理(JavaSE实现)
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实现)的更多相关文章
- 我的Android进阶之旅------>Android基于HTTP协议的多线程断点下载器的实现
一.首先写这篇文章之前,要了解实现该Android多线程断点下载器的几个知识点 1.多线程下载的原理,如下图所示 注意:由于Android移动设备和PC机的处理器还是不能相比,所以开辟的子线程建议不要 ...
- Android(java)学习笔记216:多线程断点下载的原理(Android实现)
之前在Android(java)学习笔记215中,我们从JavaSE的角度去实现了多线程断点下载,下面从Android角度实现这个断点下载: 1.新建一个Android工程: (1)其中我们先实现布局 ...
- Android(java)学习笔记159:多线程断点下载的原理(Android实现)
之前在Android(java)学习笔记215中,我们从JavaSE的角度去实现了多线程断点下载,下面从Android角度实现这个断点下载: 1. 新建一个Android工程: (1)其中我们先实现布 ...
- Android(java)学习笔记158:多线程断点下载的原理(JavaSE实现)
1. 为什么需要多线程下载? 服务器的资源有限,同时的平均地分配给每个客户端.开启的线程越多抢占的服务的资源就越多,下载的速度就越块. 2. 下载速度的限制条件? (1)你的电脑手机宽带的带宽 ...
- 【原】Java学习笔记032 - 多线程
package cn.temptation; public class Sample01 { public static void main(String[] args) { /* * [进程]:正在 ...
- Java学习笔记之——多线程
多线程编程 程序: 进程:一个程序运行就会产生一个进程 线程:进程的执行流程,一个进程至少有一个线程,称为主线程 如:QQ聊着天,同时在听音乐 一个进程可以有多个线程,多个线程共享同一个进程的资源 线 ...
- Java学习笔记:多线程(一)
Java中线程的五种状态: 新建状态(New) 就绪状态(Runnable) 运行状态(Running) 阻塞状态(Blocked) 凋亡状态(Dead) 其中阻塞状态(Blocked)又分为三种: ...
- java学习笔记(5)多线程
一.简介(过段时间再写,多线程难度有点大) --------------------------------------- 1.进程:运行时的概念,运行的应用程序 2.线程:应用程序内部并发执行的代码 ...
- Java 学习笔记(11)——多线程
Java内部提供了针对多线程的支持,线程是CPU执行的最小单位,在多核CPU中使用多线程,能够做到多个任务并行执行,提高效率. 使用多线程的方法 创建Thread类的子类,并重写run方法,在需要启动 ...
随机推荐
- Laravel PHP Web开发框架
Laravel是一套简洁.优雅的PHP Web开发框架(PHP Web Framework).它可以让你从面条一样杂乱的代码中解脱出来:它可以帮你构建一个完美的网络APP,而且每行代码都可以简洁.富于 ...
- 响应式设计的5个CSS实用技巧
正如我在教程响应式Web设计三步走当中所讲的,响应式的Web设计其实并不难,但是要让元素在布局切换时能够平滑过渡就比较考验技巧了.现在我分享在编码时常用的五个CSS技巧并举例说明.这些技巧都是使用简单 ...
- App Store生存指南
资格获取 如果已经有App Store开发帐号请跳过此节. App Store的资格获取其实一直以来都不算难,和其它事情一样,需要的只是耐心.现在苹果对申请者的文书手续要求已经比几年前简化多了 ...
- 运行在TQ2440开发板上以及X86平台上的linux内核编译
一.运行在TQ2440开发板上的linux内核编译 1.获取源码并解压 直接使用天嵌移植好的“linux-2.6.30.4_20100531.tar.bz2”源码包. 解压(天嵌默认解压到/opt/E ...
- Unable to find the ncurses libraries or the required header files解决
问题: 解决方法: sudo apt-get install ncurses-dev 参考:Unable to find the ncurses libraries or the required h ...
- HBase eclipse开发环境搭建
伪分布式环境安装 事先本机必须部署好了伪分布式hadoop开发环境将Hbase/lib下的 hadoop-core-*.jar 与 hadoop保持一致. 将hadoop下的复制到hbase下将had ...
- codeforces C. Mashmokh and Numbers
题意:给你n和k,然后让你找出n个数使得gcd(a1,a2)+gcd(a3,a4)+......的和等于k: 思路:如果n为奇数,让前n-3个数的相邻两个数都为1,n-2和n-1两个数gcd为k-an ...
- 用Java实现非阻塞通信
用ServerSocket和Socket来编写服务器程序和客户程序,是Java网络编程的最基本的方式.这些服务器程序或客户程序在运行过程中常常会阻塞.例如当一个线程执行ServerSocket的acc ...
- bzoj3531
不难想到树链剖分这题的难点是记录的是路径上宗教相同的点裸的想法是对每一种宗教都开一棵线段树,记录每个点的评级但显然这样会爆空间,仔细分析一下,这些线段树内很多点压根就没用到因此我们考虑对线段树动态开点 ...
- CH Round #48 - Streaming #3 (NOIP模拟赛Day1)
A.数三角形 题目:http://www.contesthunter.org/contest/CH%20Round%20%2348%20-%20Streaming%20%233%20(NOIP模拟赛D ...