1. 原理图

2. 示例代码

需要权限

<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>

activity_main.xml 布局文件

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context=".MainActivity" > <EditText
android:id="@+id/et_path"
android:text="http://192.168.1.100:8080/360.exe"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:hint="请输入下载的路径" >
</EditText> <ProgressBar
android:id="@+id/pb"
style="?android:attr/progressBarStyleHorizontal"
android:layout_width="fill_parent"
android:layout_height="wrap_content" /> <Button
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:onClick="downLoad"
android:text="下载" /> <TextView
android:id="@+id/tv_process"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="下载进度" /> </LinearLayout>

MainActivity.java

public class MainActivity extends Activity {
protected static final int DOWN_LOAD_ERROR = 1;
protected static final int SERVER_ERROR = 2;
public static final int DOWN_LAOD_FINSIH = 3;
public static final int UPDATE_TEXT = 4;
private EditText et_path;
private ProgressBar pb; //下载的进度条.
public static int threadCount = 3;
public static int runningThread = 3 ; public int currentProcess = 0; //当前进度. private TextView tv_process; private Handler handler = new Handler(){
public void handleMessage(android.os.Message msg) {
switch (msg.what) {
case DOWN_LOAD_ERROR:
Toast.makeText(getApplicationContext(), "下载失败", 0).show();
break;
case SERVER_ERROR:
Toast.makeText(getApplicationContext(), "服务器 错误,下载失败", 0).show();
break;
case DOWN_LAOD_FINSIH:
Toast.makeText(getApplicationContext(), "文件下载完毕", 0).show();
break;
case UPDATE_TEXT:
tv_process.setText("当前进度:"+pb.getProgress()*100/pb.getMax());
break;
} };
}; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
et_path = (EditText) this.findViewById(R.id.et_path);
pb = (ProgressBar) findViewById(R.id.pb);
tv_process = (TextView) findViewById(R.id.tv_process);
} public void downLoad(View view) {
final String path = et_path.getText().toString().trim();
if (TextUtils.isEmpty(path)) {
Toast.makeText(this, "下载路径错误", 0).show();
return;
}
currentProcess = 0;
new Thread() {
public void run() {
try {
//String path = "http://192.168.1.100:8080/360.exe";
URL url = new URL(path);
HttpURLConnection conn = (HttpURLConnection) url
.openConnection();
conn.setConnectTimeout(5000);
conn.setRequestMethod("GET");
int code = conn.getResponseCode();
if (code == 200) {
// 服务器返回的数据的长度 实际上就是文件的长度
int length = conn.getContentLength();
pb.setMax(length);//设置进度条的最大值.
System.out.println("文件总长度:" + length);
// 在客户端本地 创建出来一个大小跟服务器端文件一样大小的临时文件
RandomAccessFile raf = new RandomAccessFile("/sdcard/setup.exe",
"rwd");
// 指定创建的这个文件的长度
raf.setLength(length);
raf.close(); // 假设是3个线程去下载资源.
// 平均每一个线程下载的文件的大小.
int blockSize = length / threadCount;
for (int threadId = 1; threadId <= threadCount; threadId++) {
// 第一个线程下载的开始位置
int startIndex = (threadId - 1) * blockSize;
int endIndex = threadId * blockSize - 1;
if (threadId == threadCount) {// 最后一个线程 下载的长度 要稍微长一点
endIndex = length;
}
System.out.println("线程:" + threadId + "下载:---"
+ startIndex + "--->" + endIndex);
new DownloadThread(path, threadId, startIndex, endIndex)
.start();
} } else {
System.out.println("服务器错误.");
Message msg = new Message();
msg.what = SERVER_ERROR;
handler.sendMessage(msg);
}
} catch (Exception e) {
e.printStackTrace();
Message msg = new Message();
msg.what = DOWN_LOAD_ERROR;
handler.sendMessage(msg); } };
}.start(); } /**
* 下载文件的子线程 每一个线程 下载对应位置的文件
*
* @author Administrator
*
*/
public class DownloadThread extends Thread {
private int threadId;
private int startIndex;
private int endIndex;
private String path; /**
* @param path
* 下载文件在服务器上的路径
* @param threadId
* 线程id
* @param startIndex
* 线程下载的开始位置
* @param endIndex
* 线程下载的结束位置.
*/
public DownloadThread(String path, int threadId, int startIndex,
int endIndex) {
this.threadId = threadId;
this.startIndex = startIndex;
this.endIndex = endIndex;
this.path = path;
} @Override
public void run() {
try { // 检查是否存在 记录下载长度的文件 ,如果存在读取这个文件的数据.
//------------------替换成存数据库-------------------------
File tempFile = new File("/sdcard/"+threadId + ".txt");
if (tempFile.exists() && tempFile.length() > 0) {
FileInputStream fis = new FileInputStream(tempFile);
byte[] temp = new byte[1024];
int leng = fis.read(temp);
String downloadLen = new String(temp, 0, leng);
int downloadlenInt = Integer.parseInt(downloadLen); int alreadyDownlodint = downloadlenInt - startIndex ;
currentProcess+=alreadyDownlodint; //计算上次断点 已经下载的文件的长度. startIndex = downloadlenInt;//修改下载的真实的开始位置.
fis.close();
}
//--------------------------------------------
URL url = new URL(path);
HttpURLConnection conn = (HttpURLConnection) url
.openConnection();
conn.setRequestMethod("GET");
// 重要: 请求服务器下载部分的文件 指定文件的位置.
conn.setRequestProperty("Range", "bytes=" + startIndex + "-"
+ endIndex);
System.out.println("线程真实下载:" + threadId + "下载:---" + startIndex
+ "--->" + endIndex);
conn.setConnectTimeout(5000);
int code = conn.getResponseCode(); // 从服务器请求全部资源 200 ok
// 如果从服务器请求部分资源 206 ok
if (code == 206) {
InputStream is = conn.getInputStream();// 已经设置了 请求的位置
RandomAccessFile raf = new RandomAccessFile("/sdcard/setup.exe",
"rwd");
// 随机写文件的时候 从哪个位置开始写
raf.seek(startIndex);// 定位文件
int len = 0;
byte[] buffer = new byte[1024];
int total = 0;// 已经下载的数据长度
while ((len = is.read(buffer)) != -1) {
RandomAccessFile file = new RandomAccessFile("/sdcard/"+threadId
+ ".txt", "rwd");// 作用: 记录当前线程下载的数据长度
raf.write(buffer, 0, len);
total += len;
//System.out.println("线程:" + threadId + "total:" + total);
file.write(( total+startIndex+"").getBytes());//记录的是 下载位置.
file.close(); //更新进度条
synchronized (MainActivity.this) {
currentProcess+=len;//获取所有线程下载的总进度.
pb.setProgress(currentProcess);//更改界面上progressbar 进度条的进度
//特殊情况 progressbar progressdialog 进度条对话框 可以直接在子线程里面更新ui 内部代码 特殊处理
Message msg = Message.obtain();//复用旧的消息 避免创建新的消息
msg.what = UPDATE_TEXT;
handler.sendMessage(msg);
} }
is.close();
raf.close();
System.out.println("线程:" + threadId + "下载完毕了...");
} else {
System.out.println("线程:" + threadId + "下载失败...");
} //如何去判断应用程序已经下载完毕. } catch (Exception e) {
e.printStackTrace();
}finally{
threadFinish();
} } private synchronized void threadFinish() {
runningThread --;
if(runningThread==0){//所有的线程 已经执行完毕了.
for(int i= 1;i<=3;i++){
File file = new File("/sdcard/"+i+".txt");
file.delete();
}
System.out.println("文件下载完毕 ,删除所有的下载记录.");
Message msg = new Message();
msg.what = DOWN_LAOD_FINSIH;
handler.sendMessage(msg);
}
}
}
}

Android -- 多线程下载, 断点下载的更多相关文章

  1. Android 学习笔记之使用多线程实现断点下载...

    PS:莫名其妙的迷茫... 学习内容: 1.使用多线程实现文件下载...   多线程下载是加快下载速度的一种方式..通过开启多个线程去执行一个任务..可以使任务的执行速度变快..多线程的任务下载时常都 ...

  2. 安卓(android)之实现断点下载功能

    一.建立实体类 1.文件实体类 package com.example.zjw.myapplication.dao; import java.io.Serializable; /** * 预下载文件实 ...

  3. Java_java多线程下载-断点下载-超详细

    基本原理:利用URLConnection获取要下载文件的长度.头部等相关信息,并设置响应的头部信息.并且通过URLConnection获取输入流,将文件分成指定的块,每一块单独开辟一个线程完成数据的读 ...

  4. Android 多线程断点续传同时下载多个大文件

    最近学习在Android环境中一些网络请求方面的知识,其中有一部分是关于网络下载方面的知识.在这里解析一下自己写的demo,总结一下自己所学的知识.下图为demo的效果图,仿照一些应用下载商城在Lis ...

  5. java多线程断点下载原理(代码实例演示)

    原文:http://www.open-open.com/lib/view/open1423214229232.html 其实多线程断点下载原理,很简单的,那么我们就来先了解下,如何实现多线程的断点下载 ...

  6. java多线程下载文件和断点下载

    多线程,断点下载文件 import java.io.BufferedReader; import java.io.File; import java.io.FileInputStream; impor ...

  7. unity3d 关于断点下载和整个下载(用于更新)

    转自:http://blog.csdn.net/yutyliu/article/details/24346459 string t = ""; //整体下载 IEnumerator ...

  8. Android多线程文件下载器

    本应用实现的是输入文件的网络的地址,点击button開始下载,下载过程中有进度条和后面的文本提示进度, 下载过程中button不可点击,防止反复的下载,完成下载后会进行Toast的提示显示, 而且回复 ...

  9. Android 学习之--android多线程断点下载

    我们平时都用"迅雷"下载软件,当下载到一半的时候突然断网,下次开启的时候能够从上次下载的地方继续下载,而且下载速度很快,那么这是怎么做到的呢! 其实它的“快”其实就是多线程的下载实 ...

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

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

随机推荐

  1. 模拟hadoop-rpc通信

    一.RPC服务类 package com.css.rpc.server; import java.io.IOException; import org.apache.hadoop.HadoopIlle ...

  2. CSS冲突1 要关掉的css在项目内:【material-table】 中 checkbox 点击checkbox无法选中or取消,点击旁边才能选中or取消

    CSS优先级: !important > 行内样式 > 内嵌样式|链接外部样式(哪个在后面哪个优先级大) id选择器 > class选择器 > 元素选择器 react中好像还不 ...

  3. npm install命令对package-lock.json文件自动做了一些额外的更新

    今天我使用 npm 命令给项目安装file-saver,通过git却发现package-lock.json中除了file-saver组件之外的其他组件的记录也被改了 npm为何会自动做这些更改呢,又如 ...

  4. error: https://packages.elastic.co/GPG-KEY-elasticsearch: import read failed(2).

    安装filebeat报错: curl: (35) SSL connect errorerror: https://packages.elastic.co/GPG-KEY-elasticsearch: ...

  5. Install Haskell on Ubuntu and CentOS

    For Ubuntu: Step one: Install GHC If you don't want to install curl you can skip step 1 and just dir ...

  6. BigData Technique&&Application指南-笔记1

    1.数据的量级 传统企业数据量基本上在TB之上,大型互联网企业达到了PB以上. 2.大量不同的数据类型  结构化数据:是存储在数据库里,可以用二维表来逻辑表达数据.  半结构的非结构化数据:一般都是纯 ...

  7. MySQL临时表的简单用法(复制)

      当工作在非常大的表上时,你可能偶尔需要运行很多查询获得一个大量数据的小的子集,不是对整个表运行这些查询,而是让MySQL每次找出所需的少数记录,将记录选择到一个临时表可能更快些,然后在这些表运行查 ...

  8. JavaScript-dom3 json_str dom元素控制 模拟百度搜索

    访问关系-封装代码 html <!DOCTYPE html> <html lang="en"> <head> <meta charset= ...

  9. 如何通过包名打开手机里的APP

    目前已知的打开APP的方式有两种, 一种是通过openUrl打开,这种有一个严重的问题,即必须添加白名单,白名单之外的APP即时安装了也无法打开. 另一种就是今天的重点,通过包名打开APP.先上核心代 ...

  10. 整理一些《纸书科学计算器》的小Tips

    本文最开始是在2016年的文章 Win10应用<纸书科学计算器>更新啦! 发表之后撰写的,当时那篇文章收到了不少人点赞,应用在国内市场的日下载量也突然上涨,让我感到受宠若惊,这里要感谢Wp ...