<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="vertical" > <TextView
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="@string/down_load" /> <EditText
android:id="@+id/et"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:hint="@string/hint"
android:text="http://192.168.1.247:8080/youdao.exe" /> <ProgressBar
android:id="@+id/pb"
style="?android:attr/progressBarStyleHorizontal"
android:layout_width="fill_parent"
android:layout_height="wrap_content" /> <TextView
android:id="@+id/tv_process"
android:layout_width="fill_parent"
android:layout_height="wrap_content" /> <Button
android:id="@+id/bt"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="@string/download" /> </LinearLayout>
 /**
* 49_多线程断点下载的实现&界面的更新 DEMO
* @author dr
*
*/
public class DemoActivity extends Activity implements OnClickListener { private ProgressBar pb;
private Button bt;
private TextView tv;
private EditText et;
boolean flag = true;
boolean stopflag = false;
int total = 0; private Handler handler = new Handler() { @Override
public void handleMessage(Message msg) {
pb.setProgress(total);
// 文件总长度
int max = pb.getMax();
// total和max都是从0开始的。
if (total >= (max - 1)) {
total = max;
flag = false;
}
int result = total * 100 / max;
tv.setText("当前进度 :" + result + "%"); super.handleMessage(msg);
}
}; @Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
pb = (ProgressBar) this.findViewById(R.id.pb);
bt = (Button) this.findViewById(R.id.bt);
tv = (TextView) this.findViewById(R.id.tv_process);
et = (EditText) this.findViewById(R.id.et);
bt.setOnClickListener(this); } @Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.bt:
// 创建一个子线程 定期的更新ui
if ("开始下载".equals(bt.getText().toString())) {
bt.setText("暂停");
stopflag = false; // 开始下载
} else {
bt.setText("开始下载");
stopflag = true;
}
new Thread() {
@Override
public void run() {
super.run();
while (flag) {
try {
sleep(1000);
// 如果total > = 文件长度
Message msg = new Message();
handler.sendMessage(msg);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}.start(); // 开始执行下载的操作
String path = et.getText().toString().trim();
if ("".equals(path)) {
Toast.makeText(this, "路径不能为空", 1).show();
return;
}
try {
URL url = new URL(path);
HttpURLConnection conn = (HttpURLConnection) url
.openConnection();
conn.setRequestMethod("GET");
conn.setConnectTimeout(5000);
conn.setRequestProperty("User-Agent",
"Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1)");
int code = conn.getResponseCode();
if (code == 200) {
int len = conn.getContentLength();
RandomAccessFile file = new RandomAccessFile("/mnt/sdcard/"
+ getFilenName(path), "rwd");
// 1.设置本地文件大小跟服务器的文件大小一致
file.setLength(len);
// 设置进度条的最大值
pb.setMax(len); // 2 .假设开启3 个线程
int threadnumber = 3;
int blocksize = len / threadnumber;
/**
* 线程1 0~ blocksize 线程2 1*bolocksize ~ 2*blocksize 线程3
* 2*blocksize ~ 文件末尾
*/
for (int i = 0; i < threadnumber; i++) {
int startposition = i * blocksize;
int endpositon = (i + 1) * blocksize;
if (i == (threadnumber - 1)) {
// 最后一个线程
endpositon = len;
} DownLoadTask task = new DownLoadTask(i, path,
startposition, endpositon);
task.start();
} }
} catch (Exception e) {
Toast.makeText(this, "下载出现异常", 0).show();
e.printStackTrace();
} break;
} } class DownLoadTask extends Thread { int threadid;
String filepath;
int startposition;
int endpositon; public DownLoadTask(int threadid, String filepath, int startposition,
int endpositon) {
this.threadid = threadid;
this.filepath = filepath;
this.startposition = startposition;
this.endpositon = endpositon; } @Override
public void run() {
try {
File postionfile = new File("/mnt/sdcard/" + threadid + ".txt");
URL url = new URL(filepath);
HttpURLConnection conn = (HttpURLConnection) url
.openConnection();
System.out.println("线程" + threadid + "正在下载 " + "开始位置 : "
+ startposition + "结束位置 " + endpositon); if (postionfile.exists()) {
FileInputStream fis = new FileInputStream(postionfile);
byte[] result = StreamTool.getBytes(fis);
String str = new String(result);
if (!"".equals(str)) {
int newstartposition = Integer.parseInt(str);
if (newstartposition > startposition) {
startposition = newstartposition;
}
}
} // "Range", "bytes=2097152-4194303")
conn.setRequestProperty("Range", "bytes=" + startposition + "-"
+ endpositon);
conn.setRequestMethod("GET");
conn.setConnectTimeout(5000);
conn.setRequestProperty("User-Agent",
"Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1)");
InputStream is = conn.getInputStream();
RandomAccessFile file = new RandomAccessFile("/mnt/sdcard/"
+ getFilenName(filepath), "rwd");
// 设置 数据从文件哪个位置开始写
file.seek(startposition);
byte[] buffer = new byte[1024];
int len = 0;
// 代表当前读到的服务器数据的位置 ,同时这个值已经存储的文件的位置
int currentPostion = startposition;
// 创建一个文件对象 ,记录当前某个文件的下载位置 while ((len = is.read(buffer)) != -1) {
if (stopflag) { // 暂停下载
return;
}
file.write(buffer, 0, len); synchronized (DemoActivity.this) { // 同步当前的
total += len;
} currentPostion += len;
// 需要把currentPostion 信息给持久化到存储设备
String position = currentPostion + "";
FileOutputStream fos = new FileOutputStream(postionfile);
fos.write(position.getBytes());
fos.flush();
fos.close();
} file.close();
System.out.println("线程" + threadid + "下载完毕");
// 当线程下载完毕后 把文件删除掉
if (postionfile.exists()) {
postionfile.delete();
}
} catch (Exception e) {
e.printStackTrace();
}
super.run();
}
} public String getFilenName(String path) {
int start = path.lastIndexOf("/") + 1;
return path.substring(start, path.length());
}
}
 public class StreamTool {
/**
* 把一个inputstream里面的内容转化成一个byte[]
*/
public static byte[] getBytes(InputStream is) throws Exception{
ByteArrayOutputStream bos = new ByteArrayOutputStream();
byte[] buffer = new byte[1024];
int len = 0;
while((len = is.read(buffer))!=-1){
bos.write(buffer, 0, len);
}
is.close();
bos.flush();
byte[] result = bos.toByteArray();
System.out.println(new String(result));
return result;
}
}
     <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.INTERNET"/>

33、多线程断点下载的实现&界面的更新的更多相关文章

  1. iOS开发网络篇—大文件的多线程断点下载

    http://www.cnblogs.com/wendingding/p/3947550.html iOS开发网络篇—多线程断点下载 说明:本文介绍多线程断点下载.项目中使用了苹果自带的类,实现了同时 ...

  2. iOS开发网络篇—大文件的多线程断点下载(转)

    http://www.cnblogs.com/wendingding/p/3947550.html   iOS开发网络篇—多线程断点下载 说明:本文介绍多线程断点下载.项目中使用了苹果自带的类,实现了 ...

  3. iOS开发网络篇—多线程断点下载

    iOS开发网络篇—多线程断点下载 说明:本文介绍多线程断点下载.项目中使用了苹果自带的类,实现了同时开启多条线程下载一个较大的文件.因为实现过程较为复杂,所以下面贴出完整的代码. 实现思路:下载开始, ...

  4. andoid 多线程断点下载

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

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

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

  6. iOS开发网络请求——大文件的多线程断点下载

    iOS开发中网络请求技术已经是移动app必备技术,而网络中文件传输就是其中重点了.网络文件传输对移动客户端而言主要分为文件的上传和下载.作为开发者从技术角度会将文件分为小文件和大文件.小文件因为文件大 ...

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

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

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

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

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

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

随机推荐

  1. EXTJS 4.2 资料 控件GroupingGrid

    http://www.shuyangyang.com.cn/jishuliangongfang/qianduanjishu/2013-11-17/179.html

  2. 【BZOJ 2321】 [BeiJing2011集训]星器

    Description Magic Land上的时间又过了若干世纪…… 现在,人们谈论着一个传说:从前,他们的祖先来到了一个位于东方的岛屿,那里简直就是另外一个世界.善于分析与构造的Magic Lan ...

  3. python学习笔记27(python中sys模块的使用)

    sys.argv           命令行参数List,第一个元素是程序本身路径 sys.modules.keys() 返回所有已经导入的模块列表 sys.exc_info()     获取当前正在 ...

  4. php文件上传大小限制的修改方法大全

    php文件上传大小限制的修改方法大全 基本就是修改maxsize选项,当然为了提高上传文件的成功率,还需要设置超时时间等. 文章如下: [php文件上传]php文件上传大小限制修改,phpmyadmi ...

  5. C# 5.0 TAP 模式下的HTTP Get和Post

    标题有点瘆人,换了工作之后很少写代码了,之前由于签了保密协议,不敢把代码拿出来分享给大家,只能摘抄网上的, 今斗胆拿出来晒晒,跪求指点,直接上代码吧 public class HTTPHelper : ...

  6. js的基本概念详解

    来自<javascript高级程序设计 第三版:作者Nicholas C. Zakas>的学习笔记(三) 如果你刚学js,想快速了解到js的基本概念,以下将会是一篇不错的引导文章: 语法 ...

  7. Android中SQLite应用详解(转)

    上次我向大家介绍了SQLite的基本信息和使用过程,相信朋友们对SQLite已经有所了解了,那今天呢,我就和大家分享一下在Android中如何使用SQLite. 现在的主流移动设备像Android.i ...

  8. 分析jQuery中的each方法

    在看jQuery源码是怎么实现each方法之前,我们看一下js的原生实现. ECMAScript 5为数组定义了一个forEach方法,该方法接受两个参数:第一个参数是要在每一个数组项上运行的函数,第 ...

  9. KafkaSpout的处理流程

    基于0.93版本Storm 首先,如果自己写KafkaSpout,该怎么办?有哪些地方需要考虑呢 1. 得实现Storm指定的接口.这样Storm才能够使用它.那么需要实现什么接口?需要提供什么功能给 ...

  10. 进阶:使用 EntityManager

    JPA中要对数据库进行操作前,必须先取得EntityManager实例,这有点类似JDBC在对数据库操作之前,必须先取得Connection实例,EntityManager是JPA操作的基础,它不是设 ...