Android之——多线程下载示例(一)

转载请注明出处:http://blog.csdn.net/l1028386804/article/details/46883927

一、概述

说到Android中的文件下载,Android API中明确要求将耗时的操作放到一个子线程中执行,文件的下载无疑是需要耗费时间的,所以要将文件的下载放到子线程中执行。下面,我们一起来实现一个Android中利用多线程下载文件的小例子。

二、服务端准备

在这个小例子中我以下载有道词典为例,在网上下载有道词典的安装包,在eclipse中新建项目web,将下载的有道词典安装包放置在WebContent目录下,并将项目发布到Tomcat中,具体如下图所示

三、Android实现

1、布局

界面上自上而下放置一个TextView,用来提示文本框中输入的信息,一个文本框用来输入网络中下载文件的路径,一个Button按钮,点击下载文件,一个ProgressBar显示下载进度,一个TextView显示下载的百分比。具体布局内容如下:

<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:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    android:orientation="vertical"
    tools:context=".MainActivity" >

    <TextView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="下载路径" />

    <EditText
        android:id="@+id/ed_path"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="http://192.168.0.170:8080/web/youdao.exe"/>
    <Button
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="下载"
        android:onClick="download"/>

    <ProgressBar
        android:id="@+id/pb"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        style="@android:style/Widget.ProgressBar.Horizontal"/>

    <TextView
        android:id="@+id/tv_info"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:gravity="center"
        android:text="下载:0%"/>

</LinearLayout>

2、自定义ProgressBarListener监听器接口

新建自定义ProgressBarListener监听器接口,这个接口中定义两个方法,void getMax(int length)用来获取下载文件的长度,void getDownload(int length);用来获取每次下载的长度,这个方法中主要是在多线程中调用,子线程中获取到的数据传递到这两个接口方法中,然后在这两个接口方法中通过Handler将相应的长度信息传递到主线程,更新界面显示信息,具体代码实现如下:

package com.example.inter;

/**
 * 自定义进度条监听器
 * @author liuyazhuang
 *
 */
public interface ProgressBarListener {
	/**
	 * 获取文件的长度
	 * @param length
	 */
	void getMax(int length);
	/**
	 * 获取每次下载的长度
	 * @param length
	 */
	void getDownload(int length);
}

3、自定义线程类DownloadThread

这里通过继承Thread的方式来实现自定义线程操作,在这个类中主要是实现文件的下载操作,在这个类中,定义了一系列与下载有关的实例变量来控制下载的数据,同时通过自定义监听器ProgressBarListener中的void getDownload(int length)方法来跟新界面显示的进度信息。

具体实现如下:

package com.example.download;

import java.io.File;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL;

import com.example.inter.ProgressBarListener;

/**
 * 自定义线程类
 * @author liuyazhuang
 *
 */
public class DownloadThread extends Thread {
	//下载的线程id
	private int threadId;
	//下载的文件路径
	private String path;
	//保存的文件
	private File file;
	//下载的进度条更新的监听器
	private ProgressBarListener listener;
	//每条线程下载的数据量
	private int block;
	//下载的开始位置
	private int startPosition;
	//下载的结束位置
	private int endPosition;

	public DownloadThread(int threadId, String path, File file, ProgressBarListener listener, int block) {
		this.threadId = threadId;
		this.path = path;
		this.file = file;
		this.listener = listener;
		this.block = block;

		this.startPosition = threadId * block;
		this.endPosition = (threadId + 1) * block - 1;
	}

	@Override
	public void run() {
		super.run();
		try {
			//创建RandomAccessFile对象
			RandomAccessFile accessFile = new RandomAccessFile(file, "rwd");
			//跳转到开始位置
			accessFile.seek(startPosition);
			URL url = new URL(path);
			//打开http链接
			HttpURLConnection conn  = (HttpURLConnection) url.openConnection();
			//设置超时时间
			conn.setConnectTimeout(5000);
			//指定请求方式为GET方式
			conn.setRequestMethod("GET");
			//指定下载的位置
			conn.setRequestProperty("Range", "bytes="+startPosition + "-" + endPosition);
			//不用再去判断状态码是否为200
			InputStream in = conn.getInputStream();
			byte[] buffer = new byte[1024];
			int len = 0;
			while((len = in.read(buffer)) != -1){
				accessFile.write(buffer, 0, len);
				//更新下载进度
				listener.getDownload(len);
			}
			accessFile.close();
			in.close();
		} catch (Exception e) {
			// TODO: handle exception
			e.printStackTrace();
		}
	}
}

4、新建DownloadManager类

这个类主要是对下载过程的管理,包括下载设置下载后文件要保存的位置,计算多线程中每个线程的数据下载量等等。

具体实现如下:

package com.example.download;

import java.io.File;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL;

import android.os.Environment;

import com.example.inter.ProgressBarListener;

/**
 * 文件下载管理器
 * @author liuyazhuang
 *
 */
public class DownloadManager {
	//下载线程的数量
	private static final int TREAD_SIZE = 3;
	private File file;
	/**
	 * 下载文件的方法
	 * @param path:下载文件的路径
	 * @param listener:自定义的下载文件监听接口
	 * @throws Exception
	 */
	public void download(String path, ProgressBarListener listener) throws Exception{
		URL url = new URL(path);
		HttpURLConnection conn = (HttpURLConnection) url.openConnection();
		conn.setConnectTimeout(5000);
		conn.setRequestMethod("GET");
		if(conn.getResponseCode() == 200){
			int filesize = conn.getContentLength();
			//设置进度条的最大长度
			listener.getMax(filesize);
			//创建一个和服务器大小一样的文件
			file = new File(Environment.getExternalStorageDirectory(), this.getFileName(path));
			RandomAccessFile accessFile = new RandomAccessFile(file, "rwd");
			accessFile.setLength(filesize);
			//要关闭RandomAccessFile对象
			accessFile.close();

			//计算出每条线程下载的数据量
			int block = filesize % TREAD_SIZE == 0 ? (filesize / TREAD_SIZE) : (filesize / TREAD_SIZE +1 ); 

			//开启线程下载
			for(int i = 0; i < TREAD_SIZE; i++){
				new DownloadThread(i, path, file, listener, block).start();
			}
		}
	}

	/**
	 * 截取路径中的文件名称
	 * @param path:要截取文件名称的路径
	 * @return:截取到的文件名称
	 */
	private String getFileName(String path){
		return path.substring(path.lastIndexOf("/") + 1);
	}
}

5、完善MainActivity

在这个类中首先,找到页面中的各个控件,实现Button按钮的onClick事件,在onClick事件中开启一个线程进行下载操作,同时子线程中获取到的数据,通过handler与Message机制传递到主线程,更新界面显示。

具体实现如下:

package com.example.multi;

import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.view.Menu;
import android.view.View;
import android.widget.EditText;
import android.widget.ProgressBar;
import android.widget.TextView;
import android.widget.Toast;

import com.example.download.DownloadManager;
import com.example.inter.ProgressBarListener;

/**
 * MainActivity整个应用程序的入口
 * @author liuyazhuang
 *
 */
public class MainActivity extends Activity {

	protected static final int ERROR_DOWNLOAD = 0;
	protected static final int SET_PROGRESS_MAX = 1;
	protected static final int UPDATE_PROGRESS = 2;

	private EditText ed_path;
	private ProgressBar pb;
	private TextView tv_info;
	private DownloadManager manager;
	//handler操作
	private Handler mHandler = new Handler(){

		public void handleMessage(android.os.Message msg) {
			switch (msg.what) {
			case ERROR_DOWNLOAD:
				//提示用户下载失败
				Toast.makeText(MainActivity.this, "下载失败", Toast.LENGTH_SHORT).show();
				break;
			case SET_PROGRESS_MAX:
				//得到最大值
				int max = (Integer) msg.obj;
				//设置进度条的最大值
				pb.setMax(max);
				break;
			case UPDATE_PROGRESS:
				//获取当前下载的长度
				int currentprogress = pb.getProgress();
				//获取新下载的长度
				int len = (Integer) msg.obj;
				//计算当前总下载长度
				int crrrentTotalProgress = currentprogress + len;
				pb.setProgress(crrrentTotalProgress);

				//获取总大小
				int maxProgress = pb.getMax();
				//计算百分比
				float value = (float)currentprogress / (float)maxProgress;
				int percent = (int) (value * 100);
				//显示下载的百分比
				tv_info.setText("下载:"+percent+"%");
				break;
			default:
				break;
			}
		};
	};
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);
		this.ed_path = (EditText) super.findViewById(R.id.ed_path);
		this.pb = (ProgressBar) super.findViewById(R.id.pb);
		this.tv_info = (TextView) super.findViewById(R.id.tv_info);
		this.manager = new DownloadManager();

	}

	@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){
		final String path = ed_path.getText().toString();
		//下载
		new Thread(new Runnable() {
			@Override
			public void run() {
				// TODO Auto-generated method stub
				try {
					manager.download(path, new ProgressBarListener() {
						@Override
						public void getMax(int length) {
							// TODO Auto-generated method stub
							Message message = new Message();
							message.what = SET_PROGRESS_MAX;
							message.obj = length;
							mHandler.sendMessage(message);
						}

						@Override
						public void getDownload(int length) {
							// TODO Auto-generated method stub
							Message message = new Message();
							message.what = UPDATE_PROGRESS;
							message.obj = length;
							mHandler.sendMessage(message);
						}
					});
				} catch (Exception e) {
					// TODO: handle exception
					e.printStackTrace();
					Message message = new Message();
					message.what = ERROR_DOWNLOAD;
					mHandler.sendMessage(message);
				}
			}
		}).start();
	}
}

6、增加权限

最后,别忘了给应用授权,这里要用到Android联网授权和向SD卡中写入文件的权限。

具体实现如下:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.multi"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="18" />
	<uses-permission android:name="android.permission.INTERNET"/>
	<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>
	<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name="com.example.multi.MainActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

四、运行效果

提醒:大家可以到http://download.csdn.net/detail/l1028386804/8899957 链接来获取完整的代码示例

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-12-15 01:37:15

Android之——多线程下载示例(一)的相关文章

Android之——断点下载示例

转载请注明出处:http://blog.csdn.net/l1028386804/article/details/46897641 在上一篇博文<Android之--多线程下载示例>中,我们讲解了如何实现Android的多线程下载功能,通过将整个文件分成多个数据块,开启多个线程,让每个线程分别下载一个相应的数据块来实现多线程下载的功能.多线程下载中,可以将下载这个耗时的操作放在子线程中执行,即不阻塞主线程,又符合Android开发的设计规范. 但是当下载的过程当中突然出现手机卡死,或者网络中断

Android版多线程下载器核心代码分享

首先给大家分享多线程下载核心类: 1 package com.example.urltest; 2 3 import java.io.IOException; 4 import java.io.InputStream; 5 import java.io.RandomAccessFile; 6 import java.net.HttpURLConnection; 7 import java.net.MalformedURLException; 8 import java.net.URL; 9 im

Android 之多线程下载原理

在Android之中呢,对于多线程的操作很是平凡,所以对于多线程的理解越深,那么对于自己的程序便能够很好的运行 这也是对于Android开发是一个重要的知识点,那么我们现在来了解多线程的下载原理. android 多线程下载 多线程下载步骤: 1.本地创建一个跟服务器一样的大小一样的文件 临时文件. 2.计算分配几个线程去下载服务器上的资源 每个文件下载的位置. 3.开启线程,每一个线程下载对应的文件. 4.如果所有的线程都把自己的数据下载完成了,服务器上的资源就被下载到本地了 如图所示:(假设

Android之——多线程下载演示样例

转载请注明出处:http://blog.csdn.net/l1028386804/article/details/46883927 一.概述 说到Android中的文件下载.Android API中明白要求将耗时的操作放到一个子线程中运行,文件的下载无疑是须要耗费时间的.所以要将文件的下载放到子线程中运行. 以下,我们一起来实现一个Android中利用多线程下载文件的小样例. 二.服务端准备 在这个小样例中我下面载有道词典为例.在网上下载有道词典的安装包,在eclipse中新建项目web.将下载

Android 模拟多线程下载

以下是一个多线程下载的例子,见代码: 1.首先是布局文件 1 <?xml version="1.0" encoding="utf-8"?> 2 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 3 android:layout_width="match_parent" 4 android:layout_height

Android开发--多线程下载加断点续传

文件下载在App应用中也用到很多,一般版本更新时多要用的文件下载来进行处理,以前也有看过很多大神有过该方面的博客,今天我也自己来实践一下,写的一般,还请大家多提意见,共同进步.主要思路: 1.多线程下载: 首先通过下载总线程数来划分文件的下载区域:利用int range = fileSize / threadCount:得到每一段下载量:每一段的位置是i * range到(i + 1) * rang  - 1,注意最后一段的位置是到filesize - 1: 通过Http协议的Range字段实现

Android中多线程下载列表的封装实现(含进度反馈)

来源:http://blog.csdn.net/u011638883/article/details/17347015 实现了一下Android中的文件多线程下载模块,支持自定义线程数.断点续传.下载任务的删除,添加等功能,这里封装了一下,功能已全部实现.不过由于使用的是最简单的手动线程数组及消息通知实现,可能还存在某些小问题.笔者会在后面的使用过程中再进行优化完善.先看一下程序测试效果,这里指定了5个下载任务,以及2个下载线程,具体如下: 要运行以上Demo需要自己搭建服务器,和简单,只需要把

Java 多线程下载示例

项目结构: FileDownload.java: package com.wl.download; import java.io.File; import java.io.FileNotFoundException; import java.io.IOException; import java.io.RandomAccessFile; import java.text.DecimalFormat; import org.apache.http.HttpResponse; import org.

Android中 多线程下载原理

计算每个线程的下载起始终止位置公式如下 文件读写方式4中类型 工程源码目录 package cn.itcast.download; import java.io.File; import java.io.InputStream; import java.io.RandomAccessFile; import java.net.HttpURLConnection; import java.net.URL; public class MulThreadDownloader { public stat