package com.itheima.mutiledownloader;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.ProtocolException;
import java.net.URL;
import java.util.ArrayList;
import java.util.List; import android.os.Bundle;
import android.os.Environment;
import android.os.Handler;
import android.os.Message;
import android.renderscript.Program.TextureType;
import android.text.TextUtils;
import android.app.Activity;
import android.view.Menu;
import android.view.View;
import android.view.ViewGroup.LayoutParams;
import android.widget.EditText;
import android.widget.LinearLayout;
import android.widget.ProgressBar;
import android.widget.Toast; public class MainActivity extends Activity {
protected static final int DOWNLOAD_ERROR = ;
private static final int THREAD_DOWNLOAD_ERROR = ;
public static final int DONWLOAD_FINISH = ;
private EditText et_path;
private EditText et_number;
private LinearLayout ll_pb;
private String path;
private static int threadCount = ;
private static int blockSize = ;
private static int livingThread = ; private List<ProgressBar> pbs; private Handler handler = new Handler() {
public void handleMessage(android.os.Message msg) {
switch (msg.what) {
case DOWNLOAD_ERROR:
Toast.makeText(getApplicationContext(), "初始化下载失败", ).show();
break;
case THREAD_DOWNLOAD_ERROR:
String str = (String) msg.obj;
Toast.makeText(getApplicationContext(), str, ).show();
break;
case DONWLOAD_FINISH:
Toast.makeText(getApplicationContext(), "下载完毕", ).show();
findViewById(R.id.bt).setEnabled(true);
break;
}
};
}; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
et_path = (EditText) findViewById(R.id.et_path);
et_number = (EditText) findViewById(R.id.et_number);
ll_pb = (LinearLayout) findViewById(R.id.ll_pb); } public void click(View view) {
path = et_path.getText().toString().trim();
String strNumber = et_number.getText().toString().trim();
if (TextUtils.isEmpty(path) || TextUtils.isEmpty(strNumber)) {
Toast.makeText(this, "请输入参数", ).show();
return;
} else {
// 下载的线程的个数
threadCount = Integer.parseInt(strNumber);
pbs = new ArrayList<ProgressBar>();
ll_pb.removeAllViews();
for (int i = ; i < threadCount; i++) {
ProgressBar pb = (ProgressBar) View.inflate(
getApplicationContext(), R.layout.pb_item, null);
pb.setPadding(, , , );
ll_pb.addView(pb, LayoutParams.MATCH_PARENT, );
//在集合里面添加进度条的引用.
pbs.add(pb);
}
findViewById(R.id.bt).setEnabled(false);
new Thread() {
public void run() {
try {
URL url = new URL(path);
HttpURLConnection conn = (HttpURLConnection) url
.openConnection();
conn.setRequestMethod("GET");
conn.setConnectTimeout();
// 服务器文件的长度
int length = conn.getContentLength();
// 本地创建一个大小跟服务器一样的文件
RandomAccessFile raf = new RandomAccessFile(Environment
.getExternalStorageDirectory()
.getAbsolutePath()
+ "/" + getFileName(path), "rw");
raf.setLength(length);
System.out.println("文件的总长度:" + length);
// 计算每个线程下载的位置.
blockSize = length / threadCount;
System.out.println("每一个块的平均长度:" + blockSize);
livingThread = threadCount;
for (int i = ; i <= threadCount; i++) {
int startIndex = (i - ) * blockSize;
int endIndex = i * blockSize - ;
if (i == threadCount) {
endIndex = length;
}
//设置进度条的总进度
pbs.get(i-).setMax(endIndex-startIndex);
new Thread(
new DownLoadTask(i, startIndex, endIndex))
.start();
}
} catch (Exception e) {
e.printStackTrace();
Message msg = Message.obtain();
msg.what = DOWNLOAD_ERROR;
handler.sendMessage(msg);
}
};
}.start(); }
} /**
* 获取下载的文件的名称
*
* @param path
* @return
*/
private static String getFileName(String path) {
int start = path.lastIndexOf("/") + ;
return path.substring(start);
} private class DownLoadTask implements Runnable { private int id;
private int startIndex;
private int endIndex; public DownLoadTask(int id, int startIndex, int endIndex) {
this.id = id;
this.startIndex = startIndex;
this.endIndex = endIndex;
} @Override
public void run() {
try {
int downloadCount = ;
URL url = new URL(path);
HttpURLConnection conn = (HttpURLConnection) url
.openConnection();
conn.setRequestMethod("GET");
conn.setConnectTimeout();
// 在每次下载之前,检查是否存在已经下载大小的记录文件.
File f = new File(Environment.getExternalStorageDirectory()
.getAbsolutePath()
+ "/"
+ getFileName(path)
+ id
+ ".txt");
if (f.exists() && f.length() > ) {
FileInputStream fis = new FileInputStream(f);
BufferedReader br = new BufferedReader(
new InputStreamReader(fis));
// 已经下载的大小.
downloadCount = Integer.parseInt(br.readLine());
startIndex += downloadCount;
br.close();
fis.close();
}
conn.setRequestProperty("Range", "bytes=" + startIndex + "-"
+ endIndex);
System.out.println("线程id:" + id + " 下载的位置:" + startIndex
+ "~~~" + endIndex);
RandomAccessFile raf = new RandomAccessFile(Environment
.getExternalStorageDirectory().getAbsolutePath()
+ "/"
+ getFileName(path), "rwd");
InputStream is = conn.getInputStream();
// 指定线程下载存放文件的开始位置.
raf.seek(startIndex);
int len = ;
byte[] buffer = new byte[];
int total = ;// 记录当前线程下载的大小.
while ((len = is.read(buffer)) != -) {
File file = new File(Environment
.getExternalStorageDirectory().getAbsolutePath()
+ "/" + getFileName(path) + id + ".txt");
RandomAccessFile positionRaf = new RandomAccessFile(file,
"rwd");
raf.write(buffer, , len);
total += len;
positionRaf.write(String.valueOf(total).getBytes());
positionRaf.close();
pbs.get(id-).setProgress(total+downloadCount);
}
raf.close();
is.close();
System.out.println("线程:" + id + "下载完毕...");
} catch (Exception e) {
e.printStackTrace();
Message msg = Message.obtain();
msg.what = THREAD_DOWNLOAD_ERROR;
msg.obj = "线程:" + id + "下载失败.";
handler.sendMessage(msg);
} finally {
synchronized (MainActivity.this) {
livingThread--;
if (livingThread <= ) {
System.out.println("全部下载完毕...");
Message msg = Message.obtain();
msg.what = DONWLOAD_FINISH;
handler.sendMessage(msg);
for (int i = ; i <= threadCount; i++) {
File f = new File(Environment
.getExternalStorageDirectory()
.getAbsolutePath()
+ "/" + getFileName(path) + i + ".txt");
System.out.println(f.delete());
}
}
} }
}
}
}

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

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

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

  2. Android多线程断点下载

    到华为后,信息管理特别严格,文件不能外发.所以好久都没写博客了,今天周日,老婆非要我学习.就闲来无事,写一篇博客,呵呵-- 前段时间,项目中提到了断点下载apk并静默安装的需求.本打算用应用市场成熟的 ...

  3. Android 多线程断点下载(非原创)

    1.服务器的CPU分配给每条线程的时间片相同,服务器带宽平均分配给每条线程,所以客户端开启的线程越多,就能抢占到更多的服务器资源,这里在客户端开启多个线程来从服务器下载资源 2.fragment_ma ...

  4. Android多线程断点下载的代码流程解析

    Step 1:创建一个用来记录线程下载信息的表 创建数据库表,于是乎我们创建一个数据库的管理器类,继承SQLiteOpenHelper类 重写onCreate()与onUpgrade()方法 DBOp ...

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

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

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

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

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

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

  8. 33、多线程断点下载的实现&界面的更新

    <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android=&quo ...

  9. andoid 多线程断点下载

    本示例介绍在Android平台下通过HTTP协议实现断点续传下载. 我们编写的是Andorid的HTTP协议多线程断点下载应用程序.直接使用单线程下载HTTP文件对我们来说是一件非常简单的事.那么,多 ...

随机推荐

  1. Discuz X1.5 利用添加好友处存储xss进行蠕虫worm扩散

    Discuz X1.5 在添加好友的地方有处存储xss,借助此处xss跟用户交互可以进行蠕虫指数扩散. 位置在添加好友处 x完之后的效果 点击后触发 ok 借助此存储xss,我们进行worm传播,dz ...

  2. Jquer Ajax xmlhttp请求成功了,为什么一直在error函数里面

    转载自http://www.cnblogs.com/sky_Great/archive/2013/01/18/2866861.html 并进行整理: 今天遇到了一个极其奇怪的问题,用各种工具检查都能看 ...

  3. python之方法总结

    python的OOP的方法有3种: 1. 实例方法: 接收self参数 2. 类方法: 接收cls参数, 并要用classmethod()注册或者@classmethod注解. 3. 静态方法: 不接 ...

  4. cocos2dx中的坐标体系

    1.UI坐标系和GL坐标系 2.本地坐标与世界坐标 本地坐标是一个相对坐标,是相对于父节点或者你指明的某个节点的相对位置来说的,本地坐标的原点在参考节点的左下角 世界坐标是一个绝对的坐标,是以屏幕的左 ...

  5. 数据结构(复习)---------字符串-----KMP算法(转载)

    字符串匹配是计算机的基本任务之一. 举例来说,有一个字符串"BBC ABCDAB ABCDABCDABDE",我想知道,里面是否包含另一个字符串"ABCDABD" ...

  6. jquery中的事件

    一.事件参数   function(event){} 1.停止冒泡事件  event.stopPropagation()  <=>  return false;2.阻止默认行为  even ...

  7. WEB前端常用的测试工具

    一.QUnit 前端测试工具 QUnit是一个强大的JavaScript单元测试框架,该框架是由jQuery团队的成员所开发,并且是jQuery的官方测试套件.Qunit是Jquery的单元测试框架, ...

  8. 30个实用的Linux find命令示例

    除了在一个目录结构下查找文件这种基本的操作,你还可以用find命令实现一些实用的操作,使你的命令行之旅更加简易. 本文将介绍15种无论是于新手还是老鸟都非常有用的Linux find命令. 首先,在你 ...

  9. Java多线程——<三>简单的线程执行:Executor

    一.概述 按照<Java多线程——<一><二>>中所讲,我们要使用线程,目前都是显示的声明Thread,并调用其start()方法.多线程并行,明显我们需要声明多个 ...

  10. C#动态多线程实例

    在C#中用多线程并不难实现.它有一个命名空间:System.Threading,提供了多线程的支持. 要开启一个新线程,须要以下的初始化: ThreadStart startDownload = ne ...