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

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

一、首先写这篇文章之前,要了解实现该Android多线程断点下载器的几个知识点

 1.多线程下载的原理,如下图所示


注意:由于Android移动设备和PC机的处理器还是不能相比,所以开辟的子线程建议不要多于5条。当然现在某些高端机子的处理器能力比较强了,就可以多开辟几条子线程。

2、为了实现断点下载,采用数据库方式记录下载的进度,这样当你将该应用退出后,下次点击下载的时候,程序会去查看该下载链接是否存在下载记录,如果存在下载记录就会判断下载的进度,如何从上次下载的进度继续开始下载。

3、特别注意在主线程里不能执行一件比较耗时的工作,否则会因主线程阻塞而无法处理用户的输入事件,导致“应用无响应”错误的出现。耗时的工作应该在子线程里执行。

4、UI控件画面的重绘(更新)是由主线程负责处理的,不能在子线程中更新UI控件的值。可以采用Handler机制,在主线程创建Handler对象,在子线程发送消息给主线程所绑定的消息队列,从消息中获取UI控件的值,然后在主线程中进行UI控件的重绘(更新)工作。
5、了解HTTP协议各个头字段的含义


二、将该下载器的具体实现代码展现出来

step1、首先查看整个Android项目的结构图

                                           

step2:设计应用的UI界面   /layout/activity_main.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
	android:orientation="vertical" android:layout_width="fill_parent"
	android:layout_height="fill_parent">

    <TextView
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="@string/path" />

    <EditText
        android:id="@+id/path"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="http://192.168.1.100:8080/Hello/a.mp4" />

    <LinearLayout
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:orientation="horizontal" >

        <Button
            android:id="@+id/downloadbutton"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="@string/startbutton" />

        <Button
            android:id="@+id/stopbutton"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:enabled="false"
            android:text="@string/stopbutton" />
    </LinearLayout>

    <ProgressBar
        android:id="@+id/progressBar"
        style="?android:attr/progressBarStyleHorizontal"
        android:layout_width="fill_parent"
        android:layout_height="18dp" />

    <TextView
        android:id="@+id/resultView"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:gravity="center" />
</LinearLayout>


/values/string.xml

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <string name="action_settings">Settings</string>
    <string name="hello_world">Hello world!</string>
    <string name="app_name">多线程断点下载器_欧阳鹏编写</string>
    <string name="path">下载路径</string>
    <string name="startbutton">开始下载</string>
    <string name="success">下载完成</string>
    <string name="error">下载失败</string>
    <string name="stopbutton">停止下载</string>
    <string name="sdcarderror">SDCard不存在或者写保护</string>
</resources>


step3、程序主应用 cn.oyp.download.MainActivity.java文件

package cn.oyp.download;

import java.io.File;

import android.app.Activity;
import android.os.Bundle;
import android.os.Environment;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.ProgressBar;
import android.widget.TextView;
import android.widget.Toast;
import cn.oyp.download.downloader.DownloadProgressListener;
import cn.oyp.download.downloader.FileDownloader;

public class MainActivity extends Activity {

	/** 下载路径文本框 **/
	private EditText pathText;
	/** 下载按钮 **/
	private Button downloadButton;
	/** 停止下载按钮 **/
	private Button stopbutton;
	/** 下载进度条 **/
	private ProgressBar progressBar;
	/** 下载结果文本框,显示下载的进度值 **/
	private TextView resultView;

	/** Hanlder的作用是用于往创建Hander对象所在的线程所绑定的消息队列发送消息 **/
	private Handler handler = new UIHander();

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);

		/** 初始化各控件 **/
		pathText = (EditText) this.findViewById(R.id.path);
		downloadButton = (Button) this.findViewById(R.id.downloadbutton);
		stopbutton = (Button) this.findViewById(R.id.stopbutton);
		progressBar = (ProgressBar) this.findViewById(R.id.progressBar);
		resultView = (TextView) this.findViewById(R.id.resultView);
		/** 设置按钮的监听 **/
		ButtonClickListener listener = new ButtonClickListener();
		downloadButton.setOnClickListener(listener);
		stopbutton.setOnClickListener(listener);
	}

	/**
	 * Hanlder的作用是用于往创建Hander对象所在的线程所绑定的消息队列发送消息
	 */
	private final class UIHander extends Handler {
		public void handleMessage(Message msg) {
			switch (msg.what) {
			case 1:
				int size = msg.getData().getInt("size"); // 获取下载的进度值
				progressBar.setProgress(size); // 实时更新,设置下载进度值
				/** 计算下载的进度百分比 */
				float num = (float) progressBar.getProgress()
						/ (float) progressBar.getMax();
				int result = (int) (num * 100);
				resultView.setText(result + "%"); // 设置下载结果文本框显示下载的进度值
				// 如果进度达到了进度最大值,即下载完毕
				if (progressBar.getProgress() == progressBar.getMax()) {
					Toast.makeText(getApplicationContext(), R.string.success, 1)
							.show();// 下载成功
				}
				break;
			case -1:
				Toast.makeText(getApplicationContext(), R.string.error, 1)
						.show();// 下载出错
				break;
			}
		}
	}
	
	/**
	 * 按钮监听类
	 */
	private final class ButtonClickListener implements View.OnClickListener {
		public void onClick(View v) {
			switch (v.getId()) {
			/** 如果是下载按钮 */
			case R.id.downloadbutton:
				String path = pathText.getText().toString();// 获取下载路径
				// 判断SD卡是否存在并且可写
				if (Environment.getExternalStorageState().equals(
						Environment.MEDIA_MOUNTED)) {
					// 获取SD卡的路径
					File saveDir = Environment.getExternalStorageDirectory();
					// 开始下载的相关操作
					download(path, saveDir);
				} else {
					Toast.makeText(getApplicationContext(),
							R.string.sdcarderror, 1).show();
				}
				downloadButton.setEnabled(false);
				stopbutton.setEnabled(true);
				break;
			/** 如果是停止下载按钮 */
			case R.id.stopbutton:
				exit();// 退出下载
				downloadButton.setEnabled(true);
				stopbutton.setEnabled(false);
				break;
			}
		}

		/**
		 * UI控件画面的重绘(更新)是由主线程负责处理的,如果在子线程中更新UI控件的值,更新后的值不会重绘到屏幕上
		 * 一定要在主线程里更新UI控件的值,这样才能在屏幕上显示出来,不能在子线程中更新UI控件的值
		 * 借用Handler来传送UI控件的值到主线程去,在主线程更新UI控件的值
		 */
		private final class DownloadTask implements Runnable {
			/** 下载路径 */
			private String path;
			/** 保存路径 */
			private File saveDir;
			/** 文件下载器 */
			private FileDownloader loader;

			/**
			 * DownloadTask的构造函数
			 * 
			 * @param path
			 *            下载路径
			 * @param saveDir
			 *            保存路径
			 */
			public DownloadTask(String path, File saveDir) {
				this.path = path;
				this.saveDir = saveDir;
			}

			/**
			 * 线程主方法
			 */
			public void run() {
				try {
					/**
					 * 构建文件下载器 将下载路径,文件保存目录,下载线程数指定好
					 */
					loader = new FileDownloader(getApplicationContext(), path,
							saveDir, 5);
					progressBar.setMax(loader.getFileSize());// 设置进度条的最大刻度(即文件的总长度)
					/**
					 * DownloadProgressListener是一个接口,onDownloadSize()为未实现的方法。
					 * onDownloadSize()方法会在download方法内部被动态赋值
					 * 监听下载数量的变化,如果不需要了解实时下载的数量,可以设置为null
					 */
					loader.download(new DownloadProgressListener() {
						public void onDownloadSize(int size) {
							// 借用Handler来传送UI控件的值到主线程去,在主线程更新UI控件的值
							Message msg = new Message();
							msg.what = 1; // 对应UIHander 获得的msg.what
							msg.getData().putInt("size", size); // 将获取的值发送给handler,用于动态更新进度
							handler.sendMessage(msg);
						}
					});
				} catch (Exception e) {
					e.printStackTrace();
					// 对应UIHander 获得的msg.what
					handler.sendMessage(handler.obtainMessage(-1)); 
				}
			}

			/**
			 * 退出下载
			 */
			public void exit() {
				if (loader != null)
					loader.exit();
			}
		}

		/** end of DownloadTask */

		/**
		 * 由于用户的输入事件(点击button, 触摸屏幕....)是由主线程负责处理的,如果主线程处于工作状态,
		 * 此时用户产生的输入事件如果没能在5秒内得到处理,系统就会报“应用无响应”错误。
		 * 所以在主线程里不能执行一件比较耗时的工作,否则会因主线程阻塞而无法处理用户的输入事件,
		 * 导致“应用无响应”错误的出现。耗时的工作应该在子线程里执行。
		 */
		private DownloadTask task;

		/**
		 * 退出下载
		 */
		public void exit() {
			if (task != null)
				task.exit();
		}

		/**
		 * 下载方法,运行在主线程,负责开辟子线程完成下载操作,这操作耗时不超过1秒
		 * 
		 * @param path
		 *            下载路径
		 * @param saveDir
		 *            保存路径
		 */
		private void download(String path, File saveDir) {
			task = new DownloadTask(path, saveDir);
			new Thread(task).start();// 开辟子线程完成下载操作
		}
	}
	/** end of ButtonClickListener **/
}


文件下载器cn.oyp.download.downloader.FileDownloader.java文件

package cn.oyp.download.downloader;

import java.io.File;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.UUID;
import java.util.concurrent.ConcurrentHashMap;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import cn.oyp.download.service.FileService;

import android.content.Context;
import android.util.Log;

/**
 * 文件下载器
 */
public class FileDownloader {
	private static final String TAG = "FileDownloader";
	/** 上下文 */
	private Context context;
	/** 文件下载服务类 */
	private FileService fileService;
	/** 是否停止下载 */
	private boolean exit;
	/** 已下载文件长度 */
	private int downloadSize = 0;
	/** 原始文件长度 */
	private int fileSize = 0;
	/** 用于下载的线程数组 */
	private DownloadThread[] threads;
	/** 本地保存文件 */
	private File saveFile;
	/** 缓存各线程下载的长度 */
	private Map<Integer, Integer> data = new ConcurrentHashMap<Integer, Integer>();
	/** 每条线程下载的长度 */
	private int block;
	/** 下载路径 */
	private String downloadUrl;

	/**
	 * 获取线程数
	 */
	public int getThreadSize() {
		return threads.length;
	}

	/**
	 * 退出下载
	 */
	public void exit() {
		this.exit = true;
	}

	/**
	 * 是否退出下载
	 */
	public boolean getExit() {
		return this.exit;
	}

	/**
	 * 获取文件大小
	 */
	public int getFileSize() {
		return fileSize;
	}

	/**
	 * 累计已下载大小
	 * 该方法在具体某个线程下载的时候会被调用
	 */
	protected synchronized void append(int size) {
		downloadSize += size;
	}

	/**
	 * 更新指定线程最后下载的位置
	 * 该方法在具体某个线程下载的时候会被调用
	 * @param threadId
	 *            线程id
	 * @param pos
	 *            最后下载的位置
	 */
	protected synchronized void update(int threadId, int pos) {
		// 缓存各线程下载的长度
		this.data.put(threadId, pos);
		// 更新数据库中的各线程下载的长度
		this.fileService.update(this.downloadUrl, threadId, pos);
	}

	/**
	 * 构建文件下载器
	 * 
	 * @param downloadUrl
	 *            下载路径
	 * @param fileSaveDir
	 *            文件保存目录
	 * @param threadNum
	 *            下载线程数
	 */
	public FileDownloader(Context context, String downloadUrl,
			File fileSaveDir, int threadNum) {
		try {
			this.context = context;
			this.downloadUrl = downloadUrl;
			fileService = new FileService(this.context);
			// 根据指定的下载路径,生成URL
			URL url = new URL(this.downloadUrl);
			if (!fileSaveDir.exists())
				fileSaveDir.mkdirs();// 如果保存路径不存在,则新建一个目录
			// 根据指定的线程数来新建线程数组
			this.threads = new DownloadThread[threadNum];
			// 打开HttpURLConnection
			HttpURLConnection conn = (HttpURLConnection) url.openConnection();
			// 设置 HttpURLConnection的断开时间
			conn.setConnectTimeout(5 * 1000);
			// 设置 HttpURLConnection的请求方式
			conn.setRequestMethod("GET");
			// 设置 HttpURLConnection的接收的文件类型
			conn.setRequestProperty(
					"Accept",
					"image/gif, image/jpeg, image/pjpeg, image/pjpeg, "
							+ "application/x-shockwave-flash, application/xaml+xml, "
							+ "application/vnd.ms-xpsdocument, application/x-ms-xbap, application/x-ms-application, "
							+ "application/vnd.ms-excel, application/vnd.ms-powerpoint, application/msword, */*");
			// 设置 HttpURLConnection的接收语音
			conn.setRequestProperty("Accept-Language", "zh-CN");
			// 指定请求uri的源资源地址
			conn.setRequestProperty("Referer", downloadUrl);
			// 设置 HttpURLConnection的字符编码
			conn.setRequestProperty("Charset", "UTF-8");
			// 检查浏览页面的访问者在用什么操作系统(包括版本号)浏览器(包括版本号)和用户个人偏好
			conn.setRequestProperty(
					"User-Agent",
					"Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 5.2;"
							+ " Trident/4.0; .NET CLR 1.1.4322; .NET CLR 2.0.50727; "
							+ ".NET CLR 3.0.04506.30; .NET CLR 3.0.4506.2152;"
							+ " .NET CLR 3.5.30729)");
			conn.setRequestProperty("Connection", "Keep-Alive");
			conn.connect();
			// 打印Http协议头
			printResponseHeader(conn);
			// 如果返回的状态码为200表示正常
			if (conn.getResponseCode() == 200) {
				this.fileSize = conn.getContentLength();// 根据响应获取文件大小
				if (this.fileSize <= 0)
					throw new RuntimeException("Unkown file size ");

				String filename = getFileName(conn);// 获取文件名称
				this.saveFile = new File(fileSaveDir, filename);// 构建保存文件
				Map<Integer, Integer> logdata = fileService
						.getData(downloadUrl);// 获取下载记录
				if (logdata.size() > 0) {// 如果存在下载记录
					for (Map.Entry<Integer, Integer> entry : logdata.entrySet())
						data.put(entry.getKey(), entry.getValue());// 把各条线程已经下载的数据长度放入data中
				}
				if (this.data.size() == this.threads.length) {// 下面计算所有线程已经下载的数据总长度
					for (int i = 0; i < this.threads.length; i++) {
						this.downloadSize += this.data.get(i + 1);
					}
					print("已经下载的长度" + this.downloadSize);
				}
				// 计算每条线程下载的数据长度
				this.block = (this.fileSize % this.threads.length) == 0 ? this.fileSize
						/ this.threads.length
						: this.fileSize / this.threads.length + 1;
			} else {
				throw new RuntimeException("server no response ");
			}
		} catch (Exception e) {
			print(e.toString());
			throw new RuntimeException("don't connection this url");
		}
	}

	/**
	 * 获取文件名
	 * 
	 * @param conn
	 *            Http连接
	 */
	private String getFileName(HttpURLConnection conn) {
		String filename = this.downloadUrl.substring(this.downloadUrl
				.lastIndexOf('/') + 1);// 截取下载路径中的文件名
		// 如果获取不到文件名称
		if (filename == null || "".equals(filename.trim())) {
			// 通过截取Http协议头分析下载的文件名
			for (int i = 0;; i++) {
				String mine = conn.getHeaderField(i);
				if (mine == null)
					break;
				/**
				 * Content-disposition 是 MIME 协议的扩展,MIME 协议指示 MIME
				 * 用户代理如何显示附加的文件。
				 * Content-Disposition就是当用户想把请求所得的内容存为一个文件的时候提供一个默认的文件名
				 * 协议头中的Content-Disposition格式如下:
				 * Content-Disposition","attachment;filename=FileName.txt");
				 */
				if ("content-disposition".equals(conn.getHeaderFieldKey(i)
						.toLowerCase())) {
					// 通过正则表达式匹配出文件名
					Matcher m = Pattern.compile(".*filename=(.*)").matcher(
							mine.toLowerCase());
					// 如果匹配到了文件名
					if (m.find())
						return m.group(1);// 返回匹配到的文件名
				}
			}
			// 如果还是匹配不到文件名,则默认取一个随机数文件名
			filename = UUID.randomUUID() + ".tmp";
		}
		return filename;
	}

	/**
	 * 开始下载文件
	 * 
	 * @param listener
	 *            监听下载数量的变化,如果不需要了解实时下载的数量,可以设置为null
	 * @return 已下载文件大小
	 * @throws Exception
	 */
	public int download(DownloadProgressListener listener) throws Exception {
		try {
			RandomAccessFile randOut = new RandomAccessFile(this.saveFile, "rw");
			if (this.fileSize > 0)
				randOut.setLength(this.fileSize);
			randOut.close();
			URL url = new URL(this.downloadUrl);
			// 如果原先未曾下载或者原先的下载线程数与现在的线程数不一致
			if (this.data.size() != this.threads.length) {
				this.data.clear();// 清除原来的线程数组
				for (int i = 0; i < this.threads.length; i++) {
					this.data.put(i + 1, 0);// 初始化每条线程已经下载的数据长度为0
				}
				this.downloadSize = 0;
			}
			//循环遍历线程数组
			for (int i = 0; i < this.threads.length; i++) {
				int downLength = this.data.get(i + 1); // 获取当前线程下载的文件长度
				// 判断线程是否已经完成下载,否则继续下载
				if (downLength < this.block
						&& this.downloadSize < this.fileSize) {
					//启动线程开始下载
					this.threads[i] = new DownloadThread(this, url,
							this.saveFile, this.block, this.data.get(i + 1),
							i + 1);
					this.threads[i].setPriority(7);
					this.threads[i].start();
				} else {
					this.threads[i] = null;
				}
			}
			//如果存在下载记录,从数据库中删除它们
			fileService.delete(this.downloadUrl);
			//重新保存下载的进度到数据库
			fileService.save(this.downloadUrl, this.data);
			boolean notFinish = true;// 下载未完成
			while (notFinish) {// 循环判断所有线程是否完成下载
				Thread.sleep(900);
				notFinish = false;// 假定全部线程下载完成
				for (int i = 0; i < this.threads.length; i++) {
					if (this.threads[i] != null && !this.threads[i].isFinish()) {// 如果发现线程未完成下载
						notFinish = true;// 设置标志为下载没有完成
						// 如果下载失败,再重新下载
						if (this.threads[i].getDownLength() == -1) {
							this.threads[i] = new DownloadThread(this, url,
									this.saveFile, this.block,
									this.data.get(i + 1), i + 1);
							this.threads[i].setPriority(7);
							this.threads[i].start();
						}
					}
				}
				if (listener != null)
					listener.onDownloadSize(this.downloadSize);// 通知目前已经下载完成的数据长度
			}
			// 如果下载完成
			if (downloadSize == this.fileSize)
				fileService.delete(this.downloadUrl);// 下载完成删除记录
		} catch (Exception e) {
			print(e.toString());
			throw new Exception("file download error");
		}
		return this.downloadSize;
	}

	/**
	 * 获取Http响应头字段
	 * @param http
	 * @return
	 */
	public static Map<String, String> getHttpResponseHeader(
			HttpURLConnection http) {
		Map<String, String> header = new LinkedHashMap<String, String>();
		for (int i = 0;; i++) {
			String mine = http.getHeaderField(i);
			if (mine == null)
				break;
			header.put(http.getHeaderFieldKey(i), mine);
		}
		return header;
	}

	/**
	 * 打印Http头字段
	 * 
	 * @param http
	 */
	public static void printResponseHeader(HttpURLConnection http) {
		Map<String, String> header = getHttpResponseHeader(http);
		for (Map.Entry<String, String> entry : header.entrySet()) {
			String key = entry.getKey() != null ? entry.getKey() + ":" : "";
			print(key + entry.getValue());
		}
	}
	/**
	 * 打印信息
	 * @param msg  信息
	 */
	private static void print(String msg) {
		Log.i(TAG, msg);
	}
}

文件下载线程 cn.oyp.download.downloader.DownloadThread.java文件

package cn.oyp.download.downloader;


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

import android.util.Log;

public class DownloadThread extends Thread {
	private static final String TAG = "DownloadThread";
	/** 本地保存文件 */
	private File saveFile;
	/** 下载路径 */
	private URL downUrl;
	/** 该线程要下载的长度 */
	private int block;
	/** 线程ID */
	private int threadId = -1;
	/** 该线程已经下载的长度 */
	private int downLength;
	/** 是否下载完成*/
	private boolean finish = false;
	/** 文件下载器 */
	private FileDownloader downloader;
	/***
	 *  构造方法
	 */
	public DownloadThread(FileDownloader downloader, URL downUrl,
			File saveFile, int block, int downLength, int threadId) {
		this.downUrl = downUrl;
		this.saveFile = saveFile;
		this.block = block;
		this.downloader = downloader;
		this.threadId = threadId;
		this.downLength = downLength;
	}
	/**
	 * 线程主方法
	 */
	@Override
	public void run() {
		if (downLength < block) {// 未下载完成
			try {
				HttpURLConnection http = (HttpURLConnection) downUrl
						.openConnection();
				http.setConnectTimeout(5 * 1000);
				http.setRequestMethod("GET");
				http.setRequestProperty(
						"Accept",
						"image/gif, image/jpeg, image/pjpeg, image/pjpeg, application/x-shockwave-flash,"
								+ " application/xaml+xml, application/vnd.ms-xpsdocument, application/x-ms-xbap, "
								+ "application/x-ms-application, application/vnd.ms-excel,"
								+ " application/vnd.ms-powerpoint, application/msword, */*");
				http.setRequestProperty("Accept-Language", "zh-CN");
				http.setRequestProperty("Referer", downUrl.toString());
				http.setRequestProperty("Charset", "UTF-8");
				// 该线程开始下载位置
				int startPos = block * (threadId - 1) + downLength;
				// 该线程下载结束位置
				int endPos = block * threadId - 1;
				// 设置获取实体数据的范围
				http.setRequestProperty("Range", "bytes=" + startPos + "-"
						+ endPos);
				http.setRequestProperty(
						"User-Agent",
						"Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 5.2; Trident/4.0;"
								+ " .NET CLR 1.1.4322; .NET CLR 2.0.50727; "
								+ ".NET CLR 3.0.04506.30; .NET CLR 3.0.4506.2152; .NET CLR 3.5.30729)");
				http.setRequestProperty("Connection", "Keep-Alive");
				//获取输入流
				InputStream inStream = http.getInputStream();
				byte[] buffer = new byte[1024];
				int offset = 0;
				print("Thread " + this.threadId
						+ " start download from position " + startPos);
				/**
				 * rwd: 打开以便读取和写入,对于 "rw",还要求对文件内容的每个更新都同步写入到基础存储设备。
				 * 对于Android移动设备一定要注意同步,否则当移动设备断电的话会丢失数据
				 */
				RandomAccessFile threadfile = new RandomAccessFile(
						this.saveFile, "rwd");
				//直接移动到文件开始位置下载的
				threadfile.seek(startPos);
				while (!downloader.getExit()
						&& (offset = inStream.read(buffer, 0, 1024)) != -1) {
					threadfile.write(buffer, 0, offset);//开始写入数据到文件
					downLength += offset;	//该线程以及下载的长度增加
					downloader.update(this.threadId, downLength);//修改数据库中该线程已经下载的数据长度
					downloader.append(offset);//文件下载器已经下载的总长度增加
				}
				threadfile.close();
				inStream.close();
				print("Thread " + this.threadId + " download finish");
				this.finish = true;
			} catch (Exception e) {
				this.downLength = -1;
				print("Thread " + this.threadId + ":" + e);
			}
		}
	}

	private static void print(String msg) {
		Log.i(TAG, msg);
	}

	/**
	 * 下载是否完成
	 * 
	 * @return
	 */
	public boolean isFinish() {
		return finish;
	}

	/**
	 * 已经下载的内容大小
	 * 
	 * @return 如果返回值为-1,代表下载失败
	 */
	public long getDownLength() {
		return downLength;
	}
}

下载进度监听接口cn.oyp.download.downloader.DownloadProgressListener.java文件

package cn.oyp.download.downloader;

/**
 * 下载进度监听接口
 */
public interface DownloadProgressListener {
	/**
	 *下载的进度 
	 */
	public void onDownloadSize(int size);
}

数据库操作类 cn.oyp.download.service.DBOpenHelper.java类

package cn.oyp.download.service;

import android.content.Context;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;

public class DBOpenHelper extends SQLiteOpenHelper {
	// 数据库文件的文件名
	private static final String DBNAME = "download.db";
	// 数据库的版本号
	private static final int VERSION = 1;

	public DBOpenHelper(Context context) {
		super(context, DBNAME, null, VERSION);
	}

	@Override
	public void onCreate(SQLiteDatabase db) {
		db.execSQL("CREATE TABLE IF NOT EXISTS filedownlog (id integer primary key autoincrement, downpath varchar(100), threadid INTEGER, downlength INTEGER)");
	}

	@Override
	public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
		db.execSQL("DROP TABLE IF EXISTS filedownlog");
		onCreate(db);
	}
}

文件下载服务类cn.oyp.download.service.FileService

package cn.oyp.download.service;

import java.util.HashMap;
import java.util.Map;

import android.content.Context;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;

/**
 * 文件下载服务类
 */
public class FileService {
	private DBOpenHelper openHelper;

	public FileService(Context context) {
		openHelper = new DBOpenHelper(context);
	}

	/**
	 * 获取每条线程已经下载的文件长度
	 * 
	 * @param path
	 * @return
	 */
	public Map<Integer, Integer> getData(String path) {
		SQLiteDatabase db = openHelper.getReadableDatabase();
		Cursor cursor = db
				.rawQuery(
						"select threadid, downlength from filedownlog where downpath=?",
						new String[] { path });
		Map<Integer, Integer> data = new HashMap<Integer, Integer>();
		while (cursor.moveToNext()) {
			data.put(cursor.getInt(0), cursor.getInt(1));
		}
		cursor.close();
		db.close();
		return data;
	}

	/**
	 * 保存每条线程已经下载的文件长度
	 * 
	 * @param path
	 * @param map
	 */
	public void save(String path, Map<Integer, Integer> map) {// int threadid,
																// int position
		SQLiteDatabase db = openHelper.getWritableDatabase();
		db.beginTransaction();
		try {
			for (Map.Entry<Integer, Integer> entry : map.entrySet()) {
				db.execSQL(
						"insert into filedownlog(downpath, threadid, downlength) values(?,?,?)",
						new Object[] { path, entry.getKey(), entry.getValue() });
			}
			db.setTransactionSuccessful();
		} finally {
			db.endTransaction();
		}
		db.close();
	}

	/**
	 * 实时更新每条线程已经下载的文件长度
	 * 
	 * @param path
	 * @param map
	 */
	public void update(String path, int threadId, int pos) {
		SQLiteDatabase db = openHelper.getWritableDatabase();
		db.execSQL(
				"update filedownlog set downlength=? where downpath=? and threadid=?",
				new Object[] { pos, path, threadId });
		db.close();
	}

	/**
	 * 当文件下载完成后,删除对应的下载记录
	 * 
	 * @param path
	 */
	public void delete(String path) {
		SQLiteDatabase db = openHelper.getWritableDatabase();
		db.execSQL("delete from filedownlog where downpath=?",
				new Object[] { path });
		db.close();
	}
}

step4:AndroidManifest.xml

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

    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="17" />
    <!-- 访问Internet权限 -->
    <uses-permission android:name="android.permission.INTERNET" />
    <!-- 在SDCard中创建与删除文件权限 -->
    <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />
    <!-- 往SDCard写入数据权限 -->
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

    <application
        android:allowBackup="true"
        android:icon="@drawable/icon"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name="cn.oyp.download.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>


step5:由于便于本项目的展示,所以新建一个JSP项目,部署到Tomcat服务器上,以供下载。


step6:部署应用,观看运行效果

1、打开应用


2、点击“开始下载”


3.点击“停止下载”


4.点击“开始下载”   会继续上一次的下载进度继续下载


5.退出应用,再进应用


6、点击“开始下载”,会继续上一次退出应用的时候的下载进度继续下载,完成断点下载



7.当下载完成的时候



==================================================================================================

  作者:欧阳鹏  欢迎转载,与人分享是进步的源泉!

  转载请保留原文地址http://blog.csdn.net/ouyang_peng

==================================================================================================




读者下载源码后,会发现下载速度特别慢,有以下两种原因:

1、由于本身的网络速度的原因,不会特别快。

2、由于使用RandomAccessFile的原因,对IO操作太过于频繁。因此,我修改了DownloadThread类,修改代码如下,修改之后对速度有了点提升。在此特别感谢 pobi 读者的意见

package cn.oyp.download.downloader;

import java.io.BufferedInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;

import android.util.Log;

public class DownloadThread extends Thread {
	private static final String TAG = "DownloadThread";
	/** 本地保存文件 */
	private File saveFile;
	/** 下载路径 */
	private URL downUrl;
	/** 该线程要下载的长度 */
	private int block;
	/** 线程ID */
	private int threadId = -1;
	/** 该线程已经下载的长度 */
	private int downLength;
	/** 是否下载完成 */
	private boolean finish = false;
	/** 文件下载器 */
	private FileDownloader downloader;

	/***
	 * 构造方法
	 */
	public DownloadThread(FileDownloader downloader, URL downUrl,
			File saveFile, int block, int downLength, int threadId) {
		this.downUrl = downUrl;
		this.saveFile = saveFile;
		this.block = block;
		this.downloader = downloader;
		this.threadId = threadId;
		this.downLength = downLength;
	}

	/**
	 * 线程主方法
	 */
	@Override
	public void run() {
		if (downLength < block) {// 未下载完成
			try {
				HttpURLConnection http = (HttpURLConnection) downUrl
						.openConnection();
				http.setConnectTimeout(5 * 1000);
				http.setRequestMethod("GET");
				http.setRequestProperty(
						"Accept",
						"image/gif, image/jpeg, image/pjpeg, image/pjpeg, application/x-shockwave-flash,"
								+ " application/xaml+xml, application/vnd.ms-xpsdocument, application/x-ms-xbap, "
								+ "application/x-ms-application, application/vnd.ms-excel,"
								+ " application/vnd.ms-powerpoint, application/msword, */*");
				http.setRequestProperty("Accept-Language", "zh-CN");
				http.setRequestProperty("Referer", downUrl.toString());
				http.setRequestProperty("Charset", "UTF-8");
				// 该线程开始下载位置
				int startPos = block * (threadId - 1) + downLength;
				// 该线程下载结束位置
				int endPos = block * threadId - 1;
				// 设置获取实体数据的范围
				http.setRequestProperty("Range", "bytes=" + startPos + "-"
						+ endPos);
				http.setRequestProperty(
						"User-Agent",
						"Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 5.2; Trident/4.0;"
								+ " .NET CLR 1.1.4322; .NET CLR 2.0.50727; "
								+ ".NET CLR 3.0.04506.30; .NET CLR 3.0.4506.2152; .NET CLR 3.5.30729)");
				http.setRequestProperty("Connection", "Keep-Alive");
				/****/
				System.out.println("DownloadThread http.getResponseCode():"
						+ http.getResponseCode());
				if (http.getResponseCode() == 206) {
					/***
					 * //获取输入流 InputStream inStream = http.getInputStream();
					 * byte[] buffer = new byte[1024]; int offset = 0;
					 * print("Thread " + this.threadId +
					 * " start download from position " + startPos);
					 * 
					 * // rwd: 打开以便读取和写入,对于 "rw",还要求对文件内容的每个更新都同步写入到基础存储设备。
					 * //对于Android移动设备一定要注意同步,否则当移动设备断电的话会丢失数据 RandomAccessFile
					 * threadfile = new RandomAccessFile( this.saveFile, "rwd");
					 * //直接移动到文件开始位置下载的 threadfile.seek(startPos); while
					 * (!downloader.getExit() && (offset = inStream.read(buffer,
					 * 0, 1024)) != -1) { threadfile.write(buffer, 0,
					 * offset);//开始写入数据到文件 downLength += offset; //该线程以及下载的长度增加
					 * downloader.update(this.threadId,
					 * downLength);//修改数据库中该线程已经下载的数据长度
					 * downloader.append(offset);//文件下载器已经下载的总长度增加 }
					 * threadfile.close();
					 * 
					 * print("Thread " + this.threadId + " download finish");
					 * this.finish = true;
					 **/
					// 获取输入流
					InputStream inStream = http.getInputStream();
					BufferedInputStream bis = new BufferedInputStream(inStream);
					byte[] buffer = new byte[1024 * 4];
					int offset = 0;
					RandomAccessFile threadfile = new RandomAccessFile(
							this.saveFile, "rwd");
					// 获取RandomAccessFile的FileChannel
					FileChannel outFileChannel = threadfile.getChannel();
					// 直接移动到文件开始位置下载的
					outFileChannel.position(startPos);
					// 分配缓冲区的大小
					while (!downloader.getExit()
							&& (offset = bis.read(buffer)) != -1) {
						outFileChannel
								.write(ByteBuffer.wrap(buffer, 0, offset));// 开始写入数据到文件
						downLength += offset; // 该线程以及下载的长度增加
						downloader.update(this.threadId, downLength);// 修改数据库中该线程已经下载的数据长度
						downloader.append(offset);// 文件下载器已经下载的总长度增加
					}
					outFileChannel.close();
					threadfile.close();
					inStream.close();
					print("Thread " + this.threadId + " download finish");
					this.finish = true;
				}
			} catch (Exception e) {
				this.downLength = -1;
				print("Thread " + this.threadId + ":" + e);
			}
		}
	}

	private static void print(String msg) {
		Log.i(TAG, msg);
	}

	/**
	 * 下载是否完成
	 * 
	 * @return
	 */
	public boolean isFinish() {
		return finish;
	}

	/**
	 * 已经下载的内容大小
	 * 
	 * @return 如果返回值为-1,代表下载失败
	 */
	public long getDownLength() {
		return downLength;
	}
}

   ==================================下面看一个gif动画===========================================

    


可以查看log日志,查看多线程下载的情况

==================================================================================================

  作者:欧阳鹏  欢迎转载,与人分享是进步的源泉!

  转载请保留原文地址http://blog.csdn.net/ouyang_peng

==================================================================================================


    博客写完后,又有读者提出了修改意见,在这儿特别感谢热心的读者给的建议,下面是读者JavaLover00000 给的建议:

修改了部分代码后,具体的优化效果如下所示,修改后下载速度确实变快了很多。

修改前的效果

修改后的效果

具体修改的代码可以到以下地址进行下载:Android基于HTTP协议的多线程断点下载器的实现源码_第二次优化之后

主要是将1、buffer改为8k  
2、因为发现花费在更新数据库的时间比 read和write加起来的时间都要多一点,所以将更新数据库进度改为下载线程出现异常的时候更新单个线程进度和FileDownloader中的exit()中更新所有线程进度


代码修改的地方具体可以查看源代码中的FileDownloader.java和DownloadThread.java

==================================================================================================

  作者:欧阳鹏  欢迎转载,与人分享是进步的源泉!

  转载请保留原文地址http://blog.csdn.net/ouyang_peng

==================================================================================================


相关文章
|
2月前
|
Java Android开发 UED
🧠Android多线程与异步编程实战!告别卡顿,让应用响应如丝般顺滑!🧵
【7月更文挑战第28天】在Android开发中,确保UI流畅性至关重要。多线程与异步编程技术可将耗时操作移至后台,避免阻塞主线程。我们通常采用`Thread`类、`Handler`与`Looper`、`AsyncTask`及`ExecutorService`等进行多线程编程。
47 2
|
7天前
|
Java 数据库 Android开发
一个Android App最少有几个线程?实现多线程的方式有哪些?
本文介绍了Android多线程编程的重要性及其实现方法,涵盖了基本概念、常见线程类型(如主线程、工作线程)以及多种多线程实现方式(如`Thread`、`HandlerThread`、`Executors`、Kotlin协程等)。通过合理的多线程管理,可大幅提升应用性能和用户体验。
25 15
一个Android App最少有几个线程?实现多线程的方式有哪些?
|
17天前
HAProxy的高级配置选项-配置haproxy支持https协议及服务器动态上下线
文章介绍了如何配置HAProxy以支持HTTPS协议和实现服务器的动态上下线。
49 8
HAProxy的高级配置选项-配置haproxy支持https协议及服务器动态上下线
|
9天前
|
Java 数据库 Android开发
一个Android App最少有几个线程?实现多线程的方式有哪些?
本文介绍了Android应用开发中的多线程编程,涵盖基本概念、常见实现方式及最佳实践。主要内容包括主线程与工作线程的作用、多线程的多种实现方法(如 `Thread`、`HandlerThread`、`Executors` 和 Kotlin 协程),以及如何避免内存泄漏和合理使用线程池。通过有效的多线程管理,可以显著提升应用性能和用户体验。
29 10
|
6天前
|
监控 安全 应用服务中间件
如何配置HTTPS协议?
如何配置HTTPS协议?
15 4
|
5天前
|
安全 网络安全 数据安全/隐私保护
HTTP与HTTPS协议区别及应用场景
在互联网高速发展的今天,HTTP与HTTPS作为数据传输的基石,作用至关重要。HTTP允许客户端与服务器间传输超文本文档,但其数据传输过程未加密,存在安全隐患;HTTPS则在此基础上加入了SSL/TLS协议,实现了数据加密传输,增强了安全性,广泛应用于电子商务、网上银行、政府网站及社交媒体平台等涉及敏感信息传输的领域,有效保护了用户隐私和数据安全。随着网络安全意识提升,HTTPS正逐渐成为主流。
|
1月前
|
XML 安全 Android开发
Flutter配置Android和IOS允许http访问
Flutter配置Android和IOS允许http访问
38 3
|
1月前
|
调度 Android开发 开发者
【颠覆传统!】Kotlin协程魔法:解锁Android应用极速体验,带你领略多线程优化的无限魅力!
【8月更文挑战第12天】多线程对现代Android应用至关重要,能显著提升性能与体验。本文探讨Kotlin中的高效多线程实践。首先,理解主线程(UI线程)的角色,避免阻塞它。Kotlin协程作为轻量级线程,简化异步编程。示例展示了如何使用`kotlinx.coroutines`库创建协程,执行后台任务而不影响UI。此外,通过协程与Retrofit结合,实现了网络数据的异步加载,并安全地更新UI。协程不仅提高代码可读性,还能确保程序高效运行,不阻塞主线程,是构建高性能Android应用的关键。
37 4
|
1月前
|
安全 算法 网络协议
【在Linux世界中追寻伟大的One Piece】HTTPS协议原理
【在Linux世界中追寻伟大的One Piece】HTTPS协议原理
40 2
|
2月前
|
缓存 网络协议 算法
(二)Java网络编程之爆肝HTTP、HTTPS、TLS协议及对称与非对称加密原理!
作为一名程序员,尤其是Java程序员,那必须得了解并掌握HTTP/HTTPS相关知识。因为在如今计算机网络通信中,HTTP协议的作用功不可没,无论是日常上网追剧、冲���、亦或是接口开发、调用等,必然存在HTTP的“影子”在内。尤其对于WEB开发者而言,HTTP几乎是每天会打交道的东西。
63 10