当我们学完java中多线程的下载后,可以将它移植到我们的安卓中来,下面是具体实现源码:

DownActivity.java

package com.example.downloads;

import java.io.File;
import java.io.IOException;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;
import com.example.downloads.utils.DownLoadThread;
import android.os.Bundle;
import android.os.Environment;
import android.os.Handler;
import android.os.Message;
import android.annotation.SuppressLint;
import android.app.Activity;
import android.text.TextUtils;
import android.view.Menu;
import android.view.View;
import android.widget.EditText;
import android.widget.ProgressBar;
import android.widget.TextView;
import android.widget.Toast; public class DownActivity extends Activity {
// 声明控件
// 路径与线程数量
public EditText et_url, et_num;
// 进度条
public static ProgressBar pb_thread;
// 显示进度的操作
public TextView tv_pb;
// 线程的数量
public static int threadNum = 3;
// 每个线程负责下载的大小
public int blockSize;
public static int threadCount;// 数量
// 访问的path
public String path;
public static boolean flag = true;
// 记录进度条的值
public static int pb_count = 0;
public static Handler handler;
public static final int TEXTVALUE = 1;
public static int pb_num = 0;
public static int size = 0; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_down);
et_url = (EditText) findViewById(R.id.et_path);
et_num = (EditText) findViewById(R.id.et_threadNum);
pb_thread = (ProgressBar) findViewById(R.id.pb_down);
tv_pb = (TextView) findViewById(R.id.tv_pb);
handler = new Handler() {
@SuppressLint("HandlerLeak")
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
switch (msg.what) {
case TEXTVALUE:
System.out.println("-------" + DownActivity.pb_count
+ "//////" + DownActivity.size);
// 改变TEXTView
pb_num = (DownActivity.pb_count * 100) / DownActivity.size;
tv_pb.setText("当前进度是+" + pb_num + "%"); break; default:
break;
}
} }; } @Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.main, menu);
return true;
} public void downLoad(View v) {
DownActivity.flag = true;
DownActivity.pb_count = 0; path = et_url.getText().toString();
String threadNum_et = et_num.getText().toString(); if (TextUtils.isEmpty(path) || TextUtils.isEmpty(threadNum_et)) {
Toast.makeText(this, "不能为空", Toast.LENGTH_LONG).show();
return;
}
Toast.makeText(this, "url:" + path + "--" + threadNum_et,
Toast.LENGTH_LONG).show();
// 转换成数字
threadNum = Integer.valueOf(threadNum_et);
new Thread(new Runnable() {
@Override
public void run() {
try {
// 创建出URL对象
URL url = new URL(path);
// 创建出 HttpURLConnection对象
HttpURLConnection httpURLConnection = (HttpURLConnection) url
.openConnection(); // 设置 发请求发送的方式
httpURLConnection.setRequestMethod("GET");
// 设置请求是否超时时间
httpURLConnection.setConnectTimeout(5000);
// 设置
httpURLConnection
.setRequestProperty("User-Agent",
" Mozilla/5.0 (compatible; MSIE 10.0; Windows NT 6.2; Trident/6.0)"); // 是否响应成功
if (httpURLConnection.getResponseCode() == 200) {
// 获取文件的大小
size = httpURLConnection.getContentLength();
System.out.println("文件的大小" + size);
// 设置进度条的最大值
pb_thread.setMax(size); // 创建文件 //保存到SD卡上 // 首先判断是否拥有sdcard
if (Environment.getExternalStorageState().equals(
Environment.MEDIA_MOUNTED)) {
// 获取sdCard文件目录对象
File sdFile = Environment
.getExternalStorageDirectory();
// 创建文件对象
File file = new File(sdFile, "youdao.exe"); RandomAccessFile accessFile = new RandomAccessFile(
file, "rwd");
// 设置文件的大小
accessFile.setLength(size);
// 每个线程下载的大小
blockSize = size / threadNum;
// 开三个线程 操作此文件
for (int i = 1; i <= threadNum; i++) {
// 1 2 3
// 计算出每个线程开始的位置
int startSize = (i - 1) * blockSize;
// 结束位置
int endSize = (i) * blockSize;
// 当线程是最后一个线程的时候
if (i == threadNum) {
// 判断文件的大小是否大于计算出来的结束位置
if (size > endSize) {
// 结束位置 等于 文件的大小
endSize = size;
}
}
// 为每个线程创建一个随机的读取
RandomAccessFile threadAccessFile = new RandomAccessFile(
file, "rwd");
new Thread(new DownLoadThread(i,
threadAccessFile, startSize, endSize,
path)).start();
} } } } catch (MalformedURLException e) { e.printStackTrace();
} catch (IOException e) { e.printStackTrace();
} } }).start();
} /**
* 暂停操作
*
* @param v
*/
public void downPause(View v) {
Toast.makeText(this, "暂停", Toast.LENGTH_LONG).show(); this.flag = false; } }

DownLoadThread.java

package com.example.downloads.utils;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL; import com.example.downloads.DownActivity; import android.os.Environment; public class DownLoadThread implements Runnable { public RandomAccessFile accessFile; // 每个线程 都拥有一个accessFile的文件对象 线程1 线程2 线程3 // 线程下载文件的起始位置
public int startSize;
public int endSize; // 文件下载的path路径
public String path; public int threadId; // 线程的标识 public DownLoadThread(int threadId, RandomAccessFile accessFile,
int startSize, int endSize, String path) { this.threadId = threadId;
this.accessFile = accessFile;
this.startSize = startSize;
this.endSize = endSize;
this.path = path;
} @Override
public void run() {
// 执行run方法
try { // 创建文件到SD卡上去 // 首先判断是否拥有sdcard
if (Environment.getExternalStorageState().equals(
Environment.MEDIA_MOUNTED)) {
// 获取sdCard文件目录对象
File sdFile = Environment.getExternalStorageDirectory();
File threadFile = new File(sdFile, threadId + ".txt"); if (threadFile.exists()) { // 读取该文件的内容
// 创建文件的输入流对象
FileInputStream fis = new FileInputStream(threadFile);
// 采用工具类读取
byte data[] = StreamTools.isToData(fis);
// 转化成字符串
String threadLen = new String(data); if ((threadLen != null) && (!"".equals(threadLen))) {
startSize = Integer.valueOf(threadLen); // 解决 416bug的错误
if (startSize > endSize) {
startSize = endSize - 1;
}
} } // 创建文件 // 创建URL对象
URL url = new URL(path);
// 创建HttpURLConnection对象
HttpURLConnection httpURLConnection = (HttpURLConnection) url
.openConnection();
// 设置请求的头 httpURLConnection.setRequestMethod("GET");
// 设置请求是否超时时间
httpURLConnection.setConnectTimeout(5000);
// 设置
httpURLConnection
.setRequestProperty("User-Agent",
" Mozilla/5.0 (compatible; MSIE 10.0; Windows NT 6.2; Trident/6.0)"); // 关键的设置
httpURLConnection.setRequestProperty("Range", "bytes="
+ startSize + "-" + endSize); // 输出当前线程
System.out.println("当前线程" + threadId + " 下载开始位置:" + startSize
+ " 下载结束位置:" + endSize);
// 响应成功 // 设置随机读取文件的 开始位置
accessFile.seek(startSize);
// 获取相应流对象
InputStream is = httpURLConnection.getInputStream();
// 创建输出流对象 byte buffer[] = new byte[1024];
int len = 0;
int threadTotal = 0;// 每个线程下载后保存记录 /
while ((len = is.read(buffer)) != -1) {
accessFile.write(buffer, 0, len);
threadTotal += len;// 记录你写入的长度 //xml文件 //改变进度条:
setProgressBar(len);
// 通过文件记录文件下载的长度
FileOutputStream fos = new FileOutputStream(threadFile);
fos.write((threadTotal + "").getBytes());
fos.flush();
fos.close();
//发送handler消息
DownActivity.handler.sendEmptyMessage(DownActivity.TEXTVALUE);
if(!DownActivity.flag){
return;
} }
accessFile.close();
is.close();
System.out.println(threadId + "线程执行完毕"); // 线程操作
synchronized (DownActivity.class) {
DownActivity.threadCount++;
if (DownActivity.threadCount >= DownActivity.threadNum) {
for (int i = 1; i <= DownActivity.threadNum; i++) {
// 获取sdCard上的文件
File deleteFile = new File(sdFile, i + ".txt");
if (deleteFile.exists()) {
// 文件删除
deleteFile.delete();
}
}
}
}
}
} catch (MalformedURLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} } public synchronized void setProgressBar(int len){
DownActivity.pb_count+=len;
DownActivity.pb_thread.setProgress(DownActivity.pb_count);
} }

StreamTools.java

package com.example.downloads.utils;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream; public class StreamTools { public static byte[] isToData(InputStream is) throws IOException{
// 字节输出流
ByteArrayOutputStream bops = new ByteArrayOutputStream();
// 读取数据的缓存区
byte buffer[] = new byte[1024];
// 读取长度的记录
int len = 0;
// 循环读取
while ((len = is.read(buffer)) != -1) {
bops.write(buffer, 0, len);
}
// 把读取的内容转换成byte数组
byte data[] = bops.toByteArray(); bops.flush();
bops.close();
is.close();
return data;
}
}

strings.xml

<?xml version="1.0" encoding="utf-8"?>
<resources> <string name="app_name">downloads</string>
<string name="action_settings">Settings</string>
<string name="tv_down">文件下载的地址</string>
<string name="tv_threadNum">线程数量</string>
<string name="tv_num">0%</string>
<string name="btn_text">下载</string>
<string name="btn_pause">暂停</string>
<string name="et_path">http://172.22.64.8:8080/doudou/youdao.exe</string>
<string name="et_threadNum">3</string> </resources>

布局文件:

<RelativeLayout 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:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context=".DownActivity" > <TextView
android:id="@+id/textView1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentLeft="true"
android:layout_alignParentTop="true"
android:text="@string/tv_down" /> <EditText
android:id="@+id/et_path"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentLeft="true"
android:layout_alignParentRight="true"
android:layout_below="@+id/textView1"
android:ems="10"
android:inputType="none"
android:text="@string/et_path" > <requestFocus />
</EditText> <TextView
android:id="@+id/textView2"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentLeft="true"
android:layout_below="@+id/et_path"
android:text="@string/tv_threadNum" /> <EditText
android:id="@+id/et_threadNum"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignLeft="@+id/textView2"
android:layout_alignRight="@+id/et_path"
android:layout_below="@+id/textView2"
android:ems="10"
android:inputType="number"
android:text="@string/et_threadNum" /> <ProgressBar
android:id="@+id/pb_down"
style="?android:attr/progressBarStyleHorizontal"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignLeft="@+id/et_threadNum"
android:layout_alignRight="@+id/et_threadNum"
android:layout_below="@+id/et_threadNum"
android:layout_marginTop="14dp" /> <TextView
android:id="@+id/tv_pb"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignRight="@+id/textView1"
android:layout_below="@+id/pb_down"
android:layout_marginTop="24dp"
android:text="@string/tv_num" /> <Button
android:id="@+id/btn_down"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignLeft="@+id/pb_down"
android:layout_below="@+id/tv_pb"
android:layout_marginTop="32dp"
android:onClick="downLoad"
android:text="@string/btn_text" /> <Button
android:id="@+id/button1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignLeft="@+id/btn_down"
android:layout_below="@+id/btn_down"
android:layout_marginTop="16dp"
android:onClick="downPause"
android:text="@string/btn_pause" /> </RelativeLayout>

效果如下:

最后要注意的是别忘了在项目清单文件中加入权限:

<!-- SDCard权限 -->
<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" />

Android开发(24)---安卓中实现多线程下载(带进度条和百分比)的更多相关文章

  1. 赵雅智_android多线程下载带进度条

    progressBar说明 在某些操作的进度中的可视指示器,为用户呈现操作的进度,还它有一个次要的进度条,用来显示中间进度,如在流媒体播放的缓冲区的进度. 一个进度条也可不确定其进度.在不确定模式下, ...

  2. Java POI 3.17导出EXCEL并下载(带进度条提示)

    导出数据 共4590条 只需要 5 秒左右,性能还算可以 我们再来测试一下 50000 条的性能...

  3. Android带进度条的文件上传,使用AsyncTask异步任务

    最近项目中要做一个带进度条的上传文件的功能,学习了AsyncTask,使用起来比较方便,将几个方法实现就行,另外做了一个很简单的demo,希望能对大家有帮助,在程序中设好文件路径和服务器IP即可. A ...

  4. Android学习记录(6)—将java中的多线程下载移植到Android中(即多线程下载在Android中的使用)③

    在这一节中,我们就来讲多线程下载以及断点续传在android中怎么使用,前两节是为本节做准备的,没有看前两节的同学,最好看完前面的两篇文章再来看这篇.其实在android端的应用和java基本上是差不 ...

  5. Android学习记录(5)—在java中学习多线程下载之断点续传②

    在上一节中我们学习了在java中学习多线程下载的基本原理和基本用法,我们并没有讲多线程的断点续传,那么这一节我们就接着上一节来讲断点续传,断点续传的重要性不言而喻,可以不用重复下载,也可以节省时间,实 ...

  6. 基于Android开发的天气预报app(源码下载)

    原文:基于Android开发的天气预报app(源码下载) 基于AndroidStudio环境开发的天气app -系统总体介绍:本天气app使用AndroidStudio这个IDE工具在Windows1 ...

  7. android:异步任务asyncTask介绍及异步任务下载图片(带进度条)

    为什么要用异步任务? 在android中仅仅有在主线程才干对ui进行更新操作.而其他线程不能直接对ui进行操作 android本身是一个多线程的操作系统,我们不能把全部的操作都放在主线程中操作 .比方 ...

  8. Android UI系列-----时间、日期、Toasts和进度条Dialog

    您可以通过点击 右下角 的按钮 来对文章内容作出评价, 也可以通过左下方的 关注按钮 来关注我的博客的最新动态. 如果文章内容对您有帮助, 不要忘记点击右下角的 推荐按钮 来支持一下哦 如果您对文章内 ...

  9. iOS之UI--Quartz2D的入门应用--重绘下载圆形进度条

    *:first-child { margin-top: 0 !important; } body > *:last-child { margin-bottom: 0 !important; } ...

随机推荐

  1. MonkeyImage API 实践全记录

    1.    背景 鉴于网上使用MonkeyImage的实例除了方法sameAs外很难找到,所以本人把实践各个API的过程记录下来然自己有更感性的认识,也为往后的工作打下更好的基础.同时也和上一篇文章& ...

  2. HDU1024 Max Sum Plus Plus(DP)

    状态:d(i,j)它代表前j划分数i部并且包括第一j最佳结果时的数.g(i,j)表示前j划分数i最好的结果时,段,g(m,n)结果,需要. 本题数据较大.需採用滚动数组.注意:这题int类型就够用了, ...

  3. 使用Prism提供的类实现WPF MVVM点餐Demo

    使用Prism提供的类实现WPF MVVM点餐Demo 由于公司开发的技术需求,近期在学习MVVM模式开发WPF应用程序.进过一段时间的学习,感受到:学习MVVM模式,最好的方法就是用MVVM做几个D ...

  4. C#中float的取值范围和精度

    原文:C#中float的取值范围和精度 float类型的表现形式: 默认情况下,赋值运算符右侧的实数被视为 double. 因此,应使用后缀 f 或 F 初始化浮点型变量,如以下示例中所示: floa ...

  5. [转载]Android中WebView自适应屏幕

    webview中右下角的缩放按钮能不能去掉 settings.setDisplayZoomControls(false); //隐藏webview缩放按钮 让Webview加载的页面居中显示有我知道的 ...

  6. 基于AngularJS的前端云组件最佳实践

    AngularJS是google设计和开发的一套前端开发框架,他能帮助开发人员更便捷地进行前端开发.AngularJS是为了克服HTML在构建应用上的不足而设计的,它非常全面且简单易学习,因此Angu ...

  7. EF 关系描述

    网络上常常看到有ef 1对1 1对多等关系的描述,按照我的理解,其根本就是为了呈现出一个视图,我最近设计了一个ef关系,请大家看一看. 需求描述 在gps车辆信息管理中,有个开户需求,其根本就是三种关 ...

  8. WCF、Web API、WCF REST、Web Service 区别

    Web Service It is based on SOAP and return data in XML form. It support only HTTP protocol. It is no ...

  9. SpringMVC类型转换、数据绑定

    SpringMVC类型转换.数据绑定详解[附带源码分析] 目录 前言 属性编辑器介绍 重要接口和类介绍 部分类和接口测试 源码分析 编写自定义的属性编辑器 总结 参考资料 前言 SpringMVC是目 ...

  10. 一种解决Code Map与ReSharper冲突导致Visual Studio 2013 Peek Definition失效的办法

    更正 3月6号: 经再次测试,单独使用Code Map或ReSharper,对Visual Studio的Peek Definition无影响,但是两者一起使用时,Peek Definition失效, ...