Android之——多线程断点续传下载示例

来源:互联网 发布:音频频谱软件 编辑:程序博客网 时间:2024/06/05 22:21


Android之——多线程断点续传下载示例

原创 2015年07月15日 21:02:39

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

一、概述

在上一篇博文《Android之——多线程下载示例》中,我们讲解了如何实现Android的多线程下载功能,通过将整个文件分成多个数据块,开启多个线程,让每个线程分别下载一个相应的数据块来实现多线程下载的功能。多线程下载中,可以将下载这个耗时的操作放在子线程中执行,即不阻塞主线程,又符合Android开发的设计规范。

但是当下载的过程当中突然出现手机卡死,或者网络中断,手机电量不足关机的现象,这时,当手机可以正常使用后,如果重新下载文件,似乎不太符合大多数用户的心理期望,那如何实现当手机可以正常联网时,基于上次断网时下载的数据来下载呢?这就是所谓的断点下载了。这篇文章主要是讲解如何实现断点下载的功能。

本文讲解的Android断点下载是基于上一篇文章《Android之——多线程下载示例》,本示例是在上一示例的基础上通过在下载的过程中,将下载的信息保存到Andoid系统自带的数据库SQLite中,当手机出现异常情况而断开网络时,由于数据库中记录了上次下载的数据信息,当手机再次联网时,读取数据库中的信息,从上次断开下载的地方继续下载数据。好,不多说了,进入正文。

二、服务端准备

服务端的实现很简单,这里为了使下载的文件大些,我在网络上下载了有道词典来作为要下载的测试资源。将它放置在项目的WebContent目录下,并将项目发布在Tomcat服务器中,具体如下图所示:

就这样,服务端算是弄好了,怎么样?很简单吧?相信大家都会的!

三、Android实现

Android实现部分是本文的重点,这里我们从布局开始由浅入深慢慢讲解,这里我们通过Activity来显示程序的界面,以SQLite数据库来保存下载的信息,通过ContentProvider来操作保存的记录信息,通过Handler和Message机制将子线程中的数据传递到主线程来更新UI显示。同时通过自定义监听器来实现对UI显示更新的监听操作。

1、布局实现

布局基本上和上一博文中的布局一样,没有什么大的变动,界面上自上而下放置一个TextView,用来提示文本框中输入的信息,一个文本框用来输入网络中下载文件的路径,一个Button按钮,点击下载文件,一个ProgressBar显示下载进度,一个TextView显示下载的百分比。

具体布局内容如下:

[html] view plain copy
  1. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  2.     xmlns:tools="http://schemas.android.com/tools"  
  3.     android:layout_width="match_parent"  
  4.     android:layout_height="match_parent"  
  5.     android:paddingBottom="@dimen/activity_vertical_margin"  
  6.     android:paddingLeft="@dimen/activity_horizontal_margin"  
  7.     android:paddingRight="@dimen/activity_horizontal_margin"  
  8.     android:paddingTop="@dimen/activity_vertical_margin"  
  9.     android:orientation="vertical"  
  10.     tools:context=".MainActivity" >  
  11.   
  12.     <TextView  
  13.         android:layout_width="match_parent"  
  14.         android:layout_height="wrap_content"  
  15.         android:text="下载路径" />  
  16.       
  17.     <EditText   
  18.         android:id="@+id/ed_path"  
  19.         android:layout_width="match_parent"  
  20.         android:layout_height="wrap_content"  
  21.         android:text="http://192.168.0.170:8080/web/youdao.exe"/>  
  22.     <Button   
  23.         android:layout_width="wrap_content"  
  24.         android:layout_height="wrap_content"  
  25.         android:text="下载"  
  26.         android:onClick="download"/>  
  27.       
  28.     <ProgressBar   
  29.         android:id="@+id/pb"  
  30.         android:layout_width="match_parent"  
  31.         android:layout_height="wrap_content"  
  32.         style="@android:style/Widget.ProgressBar.Horizontal"/>  
  33.       
  34.     <TextView   
  35.         android:id="@+id/tv_info"  
  36.         android:layout_width="match_parent"  
  37.         android:layout_height="wrap_content"  
  38.         android:gravity="center"  
  39.         android:text="下载:0%"/>  
  40.   
  41. </LinearLayout>  

2、自定义ProgressBarListener监听器接口

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

具体代码实现如下:

[java] view plain copy
  1. package com.example.inter;  
  2.   
  3. /** 
  4.  * 自定义进度条监听器 
  5.  * @author liuyazhuang 
  6.  * 
  7.  */  
  8. public interface ProgressBarListener {  
  9.     /** 
  10.      * 获取文件的长度 
  11.      * @param length 
  12.      */  
  13.     void getMax(int length);  
  14.     /** 
  15.      * 获取每次下载的长度 
  16.      * @param length 
  17.      */  
  18.     void getDownload(int length);  
  19. }  

3.定义数据库的相关信息类DownloadDBHelper

在这个实例中,我们将数据库的名称定义为download.db,我们需要保存主键id,文件下载后要保存的路径,每个线程的标识id,每个线程下载的文件数据块大小,所以,在创建的数据表中共有_id, path,threadid,downloadlength,详情见下图

DownloadDBHelper实现的具体代码如下:

[java] view plain copy
  1. package com.example.db;  
  2.   
  3. import android.content.Context;  
  4. import android.database.sqlite.SQLiteDatabase;  
  5. import android.database.sqlite.SQLiteDatabase.CursorFactory;  
  6. import android.database.sqlite.SQLiteOpenHelper;  
  7.   
  8. /** 
  9.  * 数据库相关类 
  10.  * @author liuyazhuang 
  11.  * 
  12.  */  
  13. public class DownloadDBHelper extends SQLiteOpenHelper {  
  14.     /** 
  15.      * 数据库名称 
  16.      */  
  17.     private static final String NAME = "download.db";  
  18.     /** 
  19.      * 原有的构造方法 
  20.      * @param context 
  21.      * @param name 
  22.      * @param factory 
  23.      * @param version 
  24.      */  
  25.     public DownloadDBHelper(Context context, String name,  
  26.             CursorFactory factory, int version) {  
  27.         super(context, name, factory, version);  
  28.     }  
  29.     /** 
  30.      * 重载构造方法 
  31.      * @param context 
  32.      */  
  33.     public DownloadDBHelper(Context context){  
  34.         super(context, NAME, null1);  
  35.     }  
  36.       
  37.     /**  
  38.      * 创建数据库时调用 
  39.      */  
  40.     @Override  
  41.     public void onCreate(SQLiteDatabase db) {  
  42.          db.execSQL("create table download(_id integer primary key autoincrement," +  
  43.                     "path text," +  
  44.                     "threadid integer," +  
  45.                     "downloadlength integer)");  
  46.   
  47.     }  
  48.     /**  
  49.      * 更新数据库时调用 
  50.      */  
  51.     @Override  
  52.     public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {  
  53.   
  54.     }  
  55.   
  56. }  

4、创建DownloadProvider类

DownloadProvider类继承自ContentProvider,提供操作数据库的方法,在这个类中,通过UriMatcher类匹配要操作的数据库,通过DownloadDBHelper对象来得到一个具体数据库实例,来对相应的数据库进行增、删、改、查操作。

具体实现如下代码所示:

[java] view plain copy
  1. package com.example.provider;  
  2.   
  3. import com.example.db.DownloadDBHelper;  
  4.   
  5. import android.content.ContentProvider;  
  6. import android.content.ContentValues;  
  7. import android.content.UriMatcher;  
  8. import android.database.Cursor;  
  9. import android.database.sqlite.SQLiteDatabase;  
  10. import android.database.sqlite.SQLiteOpenHelper;  
  11. import android.net.Uri;  
  12.   
  13. /** 
  14.  * 自定义ContentProvider实例 
  15.  * @author liuyazhuang 
  16.  * 
  17.  */  
  18. public class DownloadProvider extends ContentProvider {  
  19.     //实例化UriMatcher对象  
  20.     private static UriMatcher matcher = new UriMatcher(UriMatcher.NO_MATCH);  
  21.     //配置访问规则  
  22.     private static final String AUTHORITY = "download";  
  23.     //自定义常量  
  24.     private static final int DOWANLOAD = 10;  
  25.     static{  
  26.         //添加匹配的规则  
  27.         matcher.addURI(AUTHORITY, "download", DOWANLOAD);  
  28.     }  
  29.     private SQLiteOpenHelper mOpenHelper;  
  30.     @Override  
  31.     public boolean onCreate() {  
  32.         mOpenHelper = new DownloadDBHelper(getContext());  
  33.         return false;  
  34.     }  
  35.   
  36.     @Override  
  37.     public Cursor query(Uri uri, String[] projection, String selection,  
  38.             String[] selectionArgs, String sortOrder) {  
  39.         // TODO Auto-generated method stub  
  40.         Cursor ret = null;  
  41.         SQLiteDatabase db = mOpenHelper.getReadableDatabase();  
  42.         int code = matcher.match(uri);  
  43.         switch (code) {  
  44.         case DOWANLOAD:  
  45.             ret = db.query("download", projection, selection, selectionArgs, nullnull, sortOrder);  
  46.             break;  
  47.   
  48.         default:  
  49.             break;  
  50.         }  
  51.         return ret;  
  52.     }  
  53.   
  54.     @Override  
  55.     public String getType(Uri uri) {  
  56.         // TODO Auto-generated method stub  
  57.         return null;  
  58.     }  
  59.   
  60.     @Override  
  61.     public Uri insert(Uri uri, ContentValues values) {  
  62.         // TODO Auto-generated method stub  
  63.         SQLiteDatabase db = mOpenHelper.getWritableDatabase();  
  64.         int code = matcher.match(uri);  
  65.         switch (code) {  
  66.         case DOWANLOAD:  
  67.             db.insert("download""_id", values);  
  68.             break;  
  69.   
  70.         default:  
  71.             break;  
  72.         }  
  73.         return null;  
  74.     }  
  75.   
  76.     @Override  
  77.     public int delete(Uri uri, String selection, String[] selectionArgs) {  
  78.         SQLiteDatabase db = mOpenHelper.getWritableDatabase();  
  79.         int code = matcher.match(uri);  
  80.         switch (code) {  
  81.         case DOWANLOAD:  
  82.             db.delete("download", selection, selectionArgs);  
  83.             break;  
  84.   
  85.         default:  
  86.             break;  
  87.         }  
  88.         return 0;  
  89.     }  
  90.   
  91.     @Override  
  92.     public int update(Uri uri, ContentValues values, String selection,  
  93.             String[] selectionArgs) {  
  94.         SQLiteDatabase db = mOpenHelper.getWritableDatabase();  
  95.         int code = matcher.match(uri);  
  96.         switch (code) {  
  97.         case DOWANLOAD:  
  98.             db.update("download", values, selection, selectionArgs);  
  99.             break;  
  100.   
  101.         default:  
  102.             break;  
  103.         }  
  104.         return 0;  
  105.     }  
  106.   
  107. }  

5、创建DownloadInfo实体类

为了使程序更加面向对象化,这里我们建立DownloadInfo实体类来对数据库中的数据进行封装,DownloadInfo实体类中的数据字段与数据库中的字段相对应

具体实现代码如下:

[java] view plain copy
  1. package com.example.domain;  
  2.   
  3. /** 
  4.  * 支持断点续传时, 
  5.  * 要保存到数据库的信息 
  6.  * @author liuyazhuang 
  7.  * 
  8.  */  
  9. public class DownloadInfo {  
  10.     //主键id  
  11.     private int _id;  
  12.     //保存路径  
  13.     private String path;  
  14.     //线程的标识id  
  15.     private String threadId;  
  16.     //下载文件的大小  
  17.     private int downloadSize;  
  18.       
  19.     public DownloadInfo() {  
  20.         super();  
  21.     }  
  22.       
  23.     public DownloadInfo(int _id, String path, String threadId, int downloadSize) {  
  24.         super();  
  25.         this._id = _id;  
  26.         this.path = path;  
  27.         this.threadId = threadId;  
  28.         this.downloadSize = downloadSize;  
  29.     }  
  30.   
  31.     public int get_id() {  
  32.         return _id;  
  33.     }  
  34.     public void set_id(int _id) {  
  35.         this._id = _id;  
  36.     }  
  37.     public String getPath() {  
  38.         return path;  
  39.     }  
  40.     public void setPath(String path) {  
  41.         this.path = path;  
  42.     }  
  43.     public String getThreadId() {  
  44.         return threadId;  
  45.     }  
  46.     public void setThreadId(String threadId) {  
  47.         this.threadId = threadId;  
  48.     }  
  49.     public int getDownloadSize() {  
  50.         return downloadSize;  
  51.     }  
  52.     public void setDownloadSize(int downloadSize) {  
  53.         this.downloadSize = downloadSize;  
  54.     }  
  55. }  

6、定义外界调用的操作数据库的方法类DownloadDao

DownloadDao类中封装了一系列操作数据库的方法,这个类不是直接操作数据库对象,而是通过ContentResolver这个对象来调用DownloadProvider中的方法来实现操作数据库的功能,这里用到了ContentResolver与ContentProvider这两个Android中非常重要的类。ContentProvider即内容提供者,主要是向外提供数据,简单理解就是一个应用程序可以通过ContentProvider向外提供操作本应用程序的接口,其他应用程序可以调用ContentProvider提供的接口来操作本应用程序的数据。ContentResolver内容接接收者,它可以接收ContentProvider的向外提供的数据。

具体代码实现如下:

[java] view plain copy
  1. package com.example.dao;  
  2.   
  3. import android.content.ContentResolver;  
  4. import android.content.ContentValues;  
  5. import android.content.Context;  
  6. import android.database.Cursor;  
  7. import android.net.Uri;  
  8.   
  9. import com.example.domain.DownloadInfo;  
  10.   
  11. /** 
  12.  * 保存下载文件信息的dao类 
  13.  * @author liuyazhuang 
  14.  * 
  15.  */  
  16. public class DownloadDao {  
  17.       
  18.     /** 
  19.      * ContentResolver对象 
  20.      */  
  21.     private ContentResolver cr;  
  22.       
  23.     public DownloadDao(Context context){  
  24.         this.cr = context.getContentResolver();  
  25.     }  
  26.     /** 
  27.      * 保存下载信息记录 
  28.      * @param info 
  29.      */  
  30.     public void save(DownloadInfo info){  
  31.         Uri uri = Uri.parse("content://download/download");  
  32.         ContentValues values = new ContentValues();  
  33.         values.put("path", info.getPath());  
  34.         values.put("threadid", info.getThreadId());  
  35.         cr.insert(uri, values);  
  36.     }  
  37.       
  38.     /** 
  39.      * 更新下载信息记录 
  40.      * @param info 
  41.      */  
  42.     public void update(DownloadInfo info){  
  43.         Uri uri = Uri.parse("content://download/download");  
  44.         ContentValues values = new ContentValues();  
  45.         values.put("downloadlength", info.getDownloadSize());  
  46.         values.put("threadid", info.getThreadId());  
  47.         cr.update(uri, values, " path = ? and threadid = ? "new String[]{info.getPath(), info.getThreadId()});  
  48.     }  
  49.     /** 
  50.      * 删除下载信息记录 
  51.      * @param info 
  52.      */  
  53.     public void delete(DownloadInfo info){  
  54.         Uri uri = Uri.parse("content://download/download");  
  55.         cr.delete(uri, " path = ? and threadid = ? "new String[]{info.getPath(), info.getThreadId()});  
  56.     }  
  57.     /** 
  58.      * 删除下载信息记录 
  59.      * @param info 
  60.      */  
  61.     public void delete(String path){  
  62.         Uri uri = Uri.parse("content://download/download");  
  63.         cr.delete(uri, " path = ? "new String[]{path});  
  64.     }  
  65.       
  66.     /** 
  67.      * 判断是否有下载记录 
  68.      * @param path 
  69.      * @return 
  70.      */  
  71.     public boolean isExist(String path){  
  72.         boolean result = false;  
  73.         Uri uri = Uri.parse("content://download/download");  
  74.         Cursor cursor = cr.query(uri, null" path = ? "new String[]{path}, null);  
  75.         if(cursor.moveToNext()){  
  76.             result = true;  
  77.         }  
  78.         cursor.close();  
  79.         return result;  
  80.     }  
  81.       
  82.     /** 
  83.      * 计算所有的下载长度 
  84.      * @param path 
  85.      * @return 
  86.      */  
  87.     public int queryCount(String path){  
  88.         int count = 0;  
  89.         Uri uri = Uri.parse("content://download/download");  
  90.         Cursor cursor = cr.query(uri, new String[]{"downloadlength"}, " path = ? "new String[]{path}, null);  
  91.         while(cursor.moveToNext()){  
  92.             int len = cursor.getInt(0);  
  93.             count += len;  
  94.         }  
  95.         cursor.close();  
  96.         return count;  
  97.     }  
  98.     /** 
  99.      * 计算每个线程的下载长度 
  100.      * @param path 
  101.      * @return 
  102.      */  
  103.     public int query(DownloadInfo info){  
  104.         int count = 0;  
  105.         Uri uri = Uri.parse("content://download/download");  
  106.         Cursor cursor = cr.query(uri, new String[]{"downloadlength"}, " path = ? and threadid = ?"new String[]{info.getPath(), info.getThreadId()}, null);  
  107.         while(cursor.moveToNext()){  
  108.             int len = cursor.getInt(0);  
  109.             count += len;  
  110.         }  
  111.         cursor.close();  
  112.         return count;  
  113.     }  
  114. }  

7、自定义线程类DownThread

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

具体实现代码如下:

[java] view plain copy
  1. package com.example.download;  
  2.   
  3. import java.io.File;  
  4. import java.io.InputStream;  
  5. import java.io.RandomAccessFile;  
  6. import java.net.HttpURLConnection;  
  7. import java.net.URL;  
  8.   
  9. import android.content.Context;  
  10.   
  11. import com.example.dao.DownloadDao;  
  12. import com.example.domain.DownloadInfo;  
  13. import com.example.inter.ProgressBarListener;  
  14.   
  15. /** 
  16.  * 自定义线程类 
  17.  * @author liuyazhuang 
  18.  * 
  19.  */  
  20. public class DownloadThread extends Thread {  
  21.     //下载的线程id  
  22.     private int threadId;  
  23.     //下载的文件路径  
  24.     private String path;  
  25.     //保存的文件  
  26.     private File file;  
  27.     //下载的进度条更新的监听器  
  28.     private ProgressBarListener listener;  
  29.     //每条线程下载的数据量  
  30.     private int block;  
  31.     //下载的开始位置  
  32.     private int startPosition;  
  33.     //下载的结束位置  
  34.     private int endPosition;  
  35.       
  36.     private DownloadDao downloadDao;  
  37.       
  38.     public DownloadThread(int threadId, String path, File file, ProgressBarListener listener, int block, Context context) {  
  39.         this.threadId = threadId;  
  40.         this.path = path;  
  41.         this.file = file;  
  42.         this.listener = listener;  
  43.         this.block = block;  
  44.         this.downloadDao = new DownloadDao(context);  
  45.         this.startPosition = threadId * block;  
  46.         this.endPosition = (threadId + 1) * block - 1;  
  47.     }  
  48.   
  49.     @Override  
  50.     public void run() {  
  51.         super.run();  
  52.         try {  
  53.             //判断该线程是否有下载记录  
  54.             DownloadInfo info = new DownloadInfo();  
  55.             info.setPath(path);  
  56.             info.setThreadId(String.valueOf(threadId));  
  57.             int length =  downloadDao.query(info);  
  58.             startPosition += length;  
  59.             //创建RandomAccessFile对象  
  60.             RandomAccessFile accessFile = new RandomAccessFile(file, "rwd");  
  61.             //跳转到开始位置  
  62.             accessFile.seek(startPosition);  
  63.             URL url = new URL(path);  
  64.             //打开http链接  
  65.             HttpURLConnection conn  = (HttpURLConnection) url.openConnection();  
  66.             //设置超时时间  
  67.             conn.setConnectTimeout(5000);  
  68.             //指定请求方式为GET方式  
  69.             conn.setRequestMethod("GET");  
  70.             //指定下载的位置  
  71.             conn.setRequestProperty("Range""bytes="+startPosition + "-" + endPosition);  
  72.             //不用再去判断状态码是否为200  
  73.             InputStream in = conn.getInputStream();  
  74.             byte[] buffer = new byte[1024];  
  75.             int len = 0;  
  76.             //该线程下载的总数据量  
  77.             int count = length;  
  78.             while((len = in.read(buffer)) != -1){  
  79.                 accessFile.write(buffer, 0, len);  
  80.                 //更新下载进度  
  81.                 listener.getDownload(len);  
  82.                 count += len;  
  83.                 info.setDownloadSize(count);  
  84.                 //更新下载的信息  
  85.                 downloadDao.update(info);  
  86.             }  
  87.             accessFile.close();  
  88.             in.close();  
  89.         } catch (Exception e) {  
  90.             // TODO: handle exception  
  91.             e.printStackTrace();  
  92.         }  
  93.     }  
  94. }  

8、新建下载的管理类DownloadManager

这个类主要是对下载过程的管理,包括下载设置下载后文件要保存的位置,计算多线程中每个线程的数据下载量等等,同时相比《Android之——多线程下载示例》一文中,它多了多下载数据的记录与更新操作。

具体实现代码如下:

[java] view plain copy
  1. package com.example.download;  
  2.   
  3. import java.io.File;  
  4. import java.io.RandomAccessFile;  
  5. import java.net.HttpURLConnection;  
  6. import java.net.URL;  
  7.   
  8. import android.content.Context;  
  9. import android.os.Environment;  
  10.   
  11. import com.example.dao.DownloadDao;  
  12. import com.example.domain.DownloadInfo;  
  13. import com.example.inter.ProgressBarListener;  
  14.   
  15. /** 
  16.  * 文件下载管理器 
  17.  * @author liuyazhuang 
  18.  * 
  19.  */  
  20. public class DownloadManager {  
  21.     //下载线程的数量  
  22.     private static final int TREAD_SIZE = 3;  
  23.     private File file;  
  24.     private DownloadDao downloadDao;  
  25.     private Context context;  
  26.     public DownloadManager(Context context) {  
  27.         this.context = context;  
  28.         this.downloadDao = new DownloadDao(context);  
  29.     }  
  30.   
  31.     /** 
  32.      * 下载文件的方法 
  33.      * @param path:下载文件的路径 
  34.      * @param listener:自定义的下载文件监听接口 
  35.      * @throws Exception 
  36.      */  
  37.     public void download(String path, ProgressBarListener listener) throws Exception{  
  38.         URL url = new URL(path);  
  39.         HttpURLConnection conn = (HttpURLConnection) url.openConnection();  
  40.         conn.setConnectTimeout(5000);  
  41.         conn.setRequestMethod("GET");  
  42.         if(conn.getResponseCode() == 200){  
  43.             int filesize = conn.getContentLength();  
  44.             //设置进度条的最大长度  
  45.             listener.getMax(filesize);  
  46.             //判断下载记录是否存在  
  47.             boolean ret = downloadDao.isExist(path);  
  48.             if(ret){  
  49.                 //得到下载的总长度,设置进度条的刻度  
  50.                 int count = downloadDao.queryCount(path);  
  51.                 listener.getDownload(count);  
  52.             }else{  
  53.                 //保存下载记录  
  54.                 for(int i = 0; i < filesize; i++){  
  55.                     DownloadInfo info = new DownloadInfo();  
  56.                     info.setPath(path);  
  57.                     info.setThreadId(String.valueOf(i));  
  58.                     //保存下载的记录信息  
  59.                     downloadDao.save(info);  
  60.                 }  
  61.             }  
  62.             //创建一个和服务器大小一样的文件  
  63.             file = new File(Environment.getExternalStorageDirectory(), this.getFileName(path));  
  64.             RandomAccessFile accessFile = new RandomAccessFile(file, "rwd");  
  65.             accessFile.setLength(filesize);  
  66.             //要关闭RandomAccessFile对象  
  67.             accessFile.close();  
  68.               
  69.             //计算出每条线程下载的数据量  
  70.             int block = filesize % TREAD_SIZE == 0 ? (filesize / TREAD_SIZE) : (filesize / TREAD_SIZE +1 );   
  71.               
  72.             //开启线程下载  
  73.             for(int i = 0; i < TREAD_SIZE; i++){  
  74.                 new DownloadThread(i, path, file, listener, block, context).start();  
  75.             }  
  76.         }  
  77.     }  
  78.       
  79.     /** 
  80.      * 截取路径中的文件名称 
  81.      * @param path:要截取文件名称的路径 
  82.      * @return:截取到的文件名称 
  83.      */  
  84.     private String getFileName(String path){  
  85.         return path.substring(path.lastIndexOf("/") + 1);  
  86.     }  
  87. }  

9、完善MainActivity

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

具体实现代码如下:

[java] view plain copy
  1. package com.example.multi;  
  2.   
  3. import android.app.Activity;  
  4. import android.os.Bundle;  
  5. import android.os.Handler;  
  6. import android.os.Message;  
  7. import android.view.Menu;  
  8. import android.view.View;  
  9. import android.widget.EditText;  
  10. import android.widget.ProgressBar;  
  11. import android.widget.TextView;  
  12. import android.widget.Toast;  
  13.   
  14. import com.example.dao.DownloadDao;  
  15. import com.example.download.DownloadManager;  
  16. import com.example.inter.ProgressBarListener;  
  17.   
  18. /** 
  19.  * MainActivity整个应用程序的入口 
  20.  * @author liuyazhuang 
  21.  * 
  22.  */  
  23. public class MainActivity extends Activity {  
  24.       
  25.     protected static final int ERROR_DOWNLOAD = 0;  
  26.     protected static final int SET_PROGRESS_MAX = 1;  
  27.     protected static final int UPDATE_PROGRESS = 2;  
  28.       
  29.     private EditText ed_path;  
  30.     private ProgressBar pb;  
  31.     private TextView tv_info;  
  32.     private DownloadManager manager;  
  33.     private DownloadDao downloadDao;  
  34.       
  35.     //handler操作  
  36.     private Handler mHandler = new Handler(){  
  37.           
  38.         public void handleMessage(android.os.Message msg) {  
  39.             switch (msg.what) {  
  40.             case ERROR_DOWNLOAD:  
  41.                 //提示用户下载失败  
  42.                 Toast.makeText(MainActivity.this"下载失败", Toast.LENGTH_SHORT).show();  
  43.                 break;  
  44.             case SET_PROGRESS_MAX:  
  45.                 //得到最大值  
  46.                 int max = (Integer) msg.obj;  
  47.                 //设置进度条的最大值  
  48.                 pb.setMax(max);  
  49.                 break;  
  50.             case UPDATE_PROGRESS:  
  51.                 //获取当前下载的长度  
  52.                 int currentprogress = pb.getProgress();  
  53.                 //获取新下载的长度  
  54.                 int len = (Integer) msg.obj;  
  55.                 //计算当前总下载长度  
  56.                 int crrrentTotalProgress = currentprogress + len;  
  57.                 pb.setProgress(crrrentTotalProgress);  
  58.                   
  59.                 //获取总大小  
  60.                 int maxProgress = pb.getMax();  
  61.                 //计算百分比  
  62.                 float value = (float)currentprogress / (float)maxProgress;  
  63.                 int percent = (int) (value * 100);  
  64.                 //显示下载的百分比  
  65.                 tv_info.setText("下载:"+percent+"%");  
  66.                   
  67.                 if(maxProgress == crrrentTotalProgress){  
  68.                     //删除下载记录  
  69.                     downloadDao.delete(ed_path.getText().toString());  
  70.                 }  
  71.                 break;  
  72.             default:  
  73.                 break;  
  74.             }  
  75.         };  
  76.     };  
  77.     @Override  
  78.     protected void onCreate(Bundle savedInstanceState) {  
  79.         super.onCreate(savedInstanceState);  
  80.         setContentView(R.layout.activity_main);  
  81.         this.ed_path = (EditText) super.findViewById(R.id.ed_path);  
  82.         this.pb = (ProgressBar) super.findViewById(R.id.pb);  
  83.         this.tv_info = (TextView) super.findViewById(R.id.tv_info);  
  84.         this.manager = new DownloadManager(this);  
  85.         this.downloadDao = new DownloadDao(this);  
  86.     }  
  87.   
  88.     @Override  
  89.     public boolean onCreateOptionsMenu(Menu menu) {  
  90.         // Inflate the menu; this adds items to the action bar if it is present.  
  91.         getMenuInflater().inflate(R.menu.main, menu);  
  92.         return true;  
  93.     }  
  94.       
  95.     public void download(View v){  
  96.         final String path = ed_path.getText().toString();  
  97.         //下载  
  98.         new Thread(new Runnable() {  
  99.             @Override  
  100.             public void run() {  
  101.                 // TODO Auto-generated method stub  
  102.                 try {  
  103.                     manager.download(path, new ProgressBarListener() {  
  104.                         @Override  
  105.                         public void getMax(int length) {  
  106.                             // TODO Auto-generated method stub  
  107.                             Message message = new Message();  
  108.                             message.what = SET_PROGRESS_MAX;  
  109.                             message.obj = length;  
  110.                             mHandler.sendMessage(message);  
  111.                         }  
  112.                           
  113.                         @Override  
  114.                         public void getDownload(int length) {  
  115.                             // TODO Auto-generated method stub  
  116.                             Message message = new Message();  
  117.                             message.what = UPDATE_PROGRESS;  
  118.                             message.obj = length;  
  119.                             mHandler.sendMessage(message);  
  120.                         }  
  121.                     });  
  122.                 } catch (Exception e) {  
  123.                     // TODO: handle exception  
  124.                     e.printStackTrace();  
  125.                     Message message = new Message();  
  126.                     message.what = ERROR_DOWNLOAD;  
  127.                     mHandler.sendMessage(message);  
  128.                 }  
  129.             }  
  130.         }).start();  
  131.     }  
  132. }  

10、增加权限

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

具体实现如下:

[html] view plain copy
  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     package="com.example.multi"  
  4.     android:versionCode="1"  
  5.     android:versionName="1.0" >  
  6.   
  7.     <uses-sdk  
  8.         android:minSdkVersion="8"  
  9.         android:targetSdkVersion="18" />  
  10.     <uses-permission android:name="android.permission.INTERNET"/>  
  11.     <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>  
  12.     <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>  
  13.     <application  
  14.         android:allowBackup="true"  
  15.         android:icon="@drawable/ic_launcher"  
  16.         android:label="@string/app_name"  
  17.         android:theme="@style/AppTheme" >  
  18.         <activity  
  19.             android:name="com.example.multi.MainActivity"  
  20.             android:label="@string/app_name" >  
  21.             <intent-filter>  
  22.                 <action android:name="android.intent.action.MAIN" />  
  23.   
  24.                 <category android:name="android.intent.category.LAUNCHER" />  
  25.             </intent-filter>  
  26.         </activity>  
  27.         <provider android:name="com.example.provider.DownloadProvider" android:authorities="download"></provider>  
  28.     </application>  
  29.   
  30. </manifest>  

四、运行效果

此时,关闭模拟器,再次打开点击下载后

如上:实现了Android中的断点下载功能。

提醒:大家可以到http://download.csdn.net/detail/l1028386804/8903201链接来获取完整的Android断点下载示例源码
原创粉丝点击