Android 学习之--android多线程断点下载

我们平时都用"迅雷"下载软件,当下载到一半的时候突然断网,下次开启的时候能够从上次下载的地方继续下载,而且下载速度很快,那么这是怎么做到的呢!

其实它的“快”其实就是多线程的下载实现的,断点下载的原理是将每次下载的字节数存取下来,保证存取的子节点跟下载的同步,并在用户下次下载的时候自动读取

存储点,并以存储点为开始值继续下载。那么android里面如何实现这么断点的下载呢?

在res的布局文件里面先画一个带有进度条的下载界面

activity_main.xml

<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:orientation="vertical"
    tools:context="com.example.thread.MainActivity" >

    <EditText
        android:id="@+id/et_path"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:hint="请输入下载路径"
        android:text="下载文件" />

    <Button
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:onClick="downLoad"
        android:text="下载" />

    <ProgressBar
        android:id="@+id/pd"
        style="?android:attr/progressBarStyleHorizontal"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content" />

    <TextView
        android:id="@+id/tv"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="下载进度" />

</LinearLayout>
package com.example.thread;

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

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

public class MainActivity extends Activity {
    private static EditText et_path;
    private static ProgressBar pb;
    private static TextView tv;
    public static int runningThread = 3;
    protected static final int DOWN_LOAD_ERROR = 1;
    protected static final int SERVER_ERROR = 2;
    protected static final int DOWN_LOAD_FINSIH = 3;
    protected static final int SHOW_TEXT = 4;
    public static int currentProcess = 0;// 当前进度
    private Handler handler = new Handler() {
        public void handleMessage(android.os.Message msg) {
            switch (msg.what) {
            case DOWN_LOAD_ERROR:
                Toast.makeText(getApplicationContext(), "下载失败", 0).show();
                break;
            case SERVER_ERROR:
                Toast.makeText(getApplicationContext(), "服务器异常", 0).show();
                break;
            case DOWN_LOAD_FINSIH:
                Toast.makeText(getApplicationContext(), "文件下载完毕", 0).show();
                break;
            case SHOW_TEXT:
                Toast.makeText(getApplicationContext(),
                        "下载进度" + pb.getProgress() * 100 / pb.getMax() + "%", 0)
                        .show();
                break;
            default:
                break;
            }
        };
    };

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        et_path = (EditText) findViewById(R.id.et_path);
        pb = (ProgressBar) findViewById(R.id.pd);
        tv = (TextView) findViewById(R.id.tv);
    }

    public void downLoad(View view) {
        String path = et_path.getText().toString().trim();
        final int thread = 3;
        if (TextUtils.isEmpty(path)) {
            Toast.makeText(this, "路径读取失败", 0).show();
            return;
        }
        new Thread() {
            public void run() {
                try {
                    // 1:连接服务器,获取一个文件,获取文件的长度,在本地创建一个大小跟服务器文件大小
                    String path = "http://dldx.csdn.net/fd.php?i=545080895071440&s=28f87f3d754b743fc81fb8b82848dcd1";
                    URL url = new URL(path);
                    HttpURLConnection conn = (HttpURLConnection) url
                            .openConnection();
                    conn.setConnectTimeout(5000);
                    conn.setRequestMethod("GET");
                    int code = conn.getResponseCode();
                    if (code == 200) {
                        // 服务器返回的数据的长度
                        int length = conn.getContentLength();
                        pb.setMax(length);// 设置进度条的最大值
                        System.out.println("文件总长度" + length);
                        // 在客户端本地创建出来一个大小跟服务器端文件大小的临时文件
                        RandomAccessFile raFile = new RandomAccessFile(
                                "/sdcard/spider_demo.rar", "rwd");
                        // 指定创建的这个文件的长度
                        raFile.setLength(length);
                        raFile.close();
                        // 平均每一个线程下载的文件的大小
                        int blockSize = length / thread;
                        for (int i = 1; i <= thread - 1; i++) {
                            // 第一个线程下载的开始位置
                            int startIndex = (i - 1) * blockSize;
                            int endIndex = i * blockSize - 1;
                            if (i == thread) {// 最后一个线程的长度要稍微长一点
                                endIndex = length;
                            }
                            System.out.println("线程:" + i + "下载:----"
                                    + startIndex + "----->" + endIndex);
                            new DownLoadthread(i, startIndex, endIndex, path)
                                    .start();
                        }
                    } else {
                        Message msg = new Message();
                        msg.what = SERVER_ERROR;
                        handler.sendMessage(msg);
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                    Message msg = new Message();
                    msg.what = DOWN_LOAD_ERROR;
                    handler.sendMessage(msg);
                }
            };

        }.start();
    }

    /**
     * 下载文件的子线程,每一个线程下载对应位置的文件
     *
     * */
    public class DownLoadthread extends Thread {
        private int threadId;
        private int startIndex;
        private int endIndex;
        private String path;

        /**
         * @param threadId线程id
         * @param startIndex
         *            线程下载的开始位置
         * @param endIndex
         *            线程下载的结束位置
         * @param path下载文件在服务器上的路径
         *
         * */
        public DownLoadthread(int threadId, int startIndex, int endIndex,
                String path) {
            this.threadId = threadId;
            this.startIndex = startIndex;
            this.endIndex = endIndex;
            this.path = path;
        }

        @Override
        public void run() {
            try {
                // 检查是否存在记录下载的文件,如果存在读取这个文件的数据
                // ----------------------替换成数据库----------------------------
                File tempFile = new File("/sdcard/" + threadId + ".txt");
                if (tempFile.exists() && tempFile.length() > 0) {
                    FileInputStream fis = new FileInputStream(tempFile);
                    byte[] temp = new byte[1024];
                    int leng = fis.read(temp);
                    String downloadLen = new String(temp, 0, leng);
                    int downloadInt = Integer.parseInt(downloadLen);
                    startIndex += downloadInt;// 修改下载的真正的开始位置
                    fis.close();
                }
                // ----------------------替换成数据库----------------------------
                URL url = new URL(path);
                HttpURLConnection conn = (HttpURLConnection) url
                        .openConnection();
                conn.setConnectTimeout(5000);
                conn.setRequestMethod("GET");
                // 重要:请求服务器下载部分的文件,需要置顶文件的位置
                conn.setRequestProperty("Range", "bytes=" + startIndex + "-"
                        + endIndex);
                // 从服务器请求全部资源,如果从服务器请求部分资源为206
                int code = conn.getResponseCode();
                System.out.println("code=" + code);
                /*
                 * if (code == 200) { InputStream is =
                 * conn.getInputStream();//返回全部的资源 }
                 */
                InputStream is = conn.getInputStream();// 返回部分的资源
                RandomAccessFile raFile = new RandomAccessFile(
                        "/sdcard/spider_demo.rar", "rwd");
                // 随机写文件的时候,从哪个文件开始写
                raFile.seek(startIndex);
                int len = 0;
                byte[] buffer = new byte[1024];
                int total = 0;// 已经下载的数据的长度
                File file = new File("/sdcard/" + threadId + ".txt");// 作用:记录当前线程下载的数据长度
                while ((len = is.read(buffer)) != -1) {
                    // 如果你是固态硬盘,那么必须这样写数据
                    // RandomAccessFile info = new RandomAccessFile(
                    // threadId + ".txt", "rwd");
                    FileOutputStream fos = new FileOutputStream(file);
                    raFile.write(buffer, 0, len);
                    total += len;
                    fos.write(String.valueOf(total).getBytes());
                    fos.close();
                    // 更新进度条
                    synchronized (MainActivity.this) {
                        currentProcess += len;// 获取所以进程下载的总进度总进度
                        pb.setProgress(currentProcess);// 更该界面上progressbar进度条的进度
                        Message message = Message.obtain();// 返回一个消息实例,防止创建太多对象
                        message.what = SHOW_TEXT;
                        // message.obj
                        handler.sendMessage(message);
                        // 特殊情况,progressbar
                        // progressdialog进度条对话框可以直接在子线程里面更新界面,它内部处理过了
                    }
                }
                is.close();
                raFile.close();
                System.out.println("线程:" + threadId + "下载完毕了...");
                File deleteFile = new File("/sdcard/" + threadId + ".txt");
                deleteFile.delete();
            } catch (Exception e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            } finally {
                threadFinish();
            }
        }

        private synchronized void threadFinish() {
            runningThread--;
            if (runningThread == 0) {
                for (int i = 1; i <= 3; i++) {
                    File file = new File("/sdcard/" + i + ".txt");
                    file.delete();
                }
                System.out.println("文件下载完毕,删除所以的下载记录");
                Message msg = new Message();
                msg.what = DOWN_LOAD_FINSIH;
                handler.sendMessage(msg);
            }
        }
    }
}

在清单文件里面加入权限

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

    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="21" />
    <uses-permission android:name="android.permission.INTERNET"/>
    <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=".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>

最终显示断点下载

时间: 2024-10-09 17:10:07

Android 学习之--android多线程断点下载的相关文章

Android/java http多线程断点下载(附源码)

先看下项目结构: http多线程断点下载涉及到 数据库,多线程和http请求等几个模块,东西不是很多,想弄清楚也不是很困难,接下来我和大家分享下我的做法. 一.先看MainActivity.java 成员变量,主要是一些下载过程的变量和handler private String path = "http://192.168.1.3:8080/wanmei/yama.apk"; private String sdcardPath; private int threadNum = 5;

Android实现多线程断点下载

本案例在于实现文件的多线程断点下载,即文件在下载一部分中断后,可继续接着已有进度下载,并通过进度条显示进度.也就是说在文件开始下载的同时,自动创建每个线程的下载进度的本地文件,下载中断后,重新进入应用点击下载,程序检查有没有本地文件的存在,若存在,获取本地文件中的下载进度,继续进行下载,当下载完成后,自动删除本地文件. 1. 定义布局文件需要用到的属性名及内容 2. 设置用户的Internet权限和关于SD卡的权限 <span style="font-size:14px;">

Android 多线程断点下载源码

源码下载地址   http://download.csdn.net/detail/kiduo08/7709391 Android 多线程断点下载源码

[Android学习笔记]Android中多线程开发的一些概念

线程安全: 在多线程的情况下,不会因为线程之间的操作而导致数据错误. 线程同步: 同一个资源,可能在同一时间被多个线程操作,这样会导致数据错误.这是一个现象,也是一个问题,而研究如何解决此类问题的相关工作就叫做线程同步. android中,处理线程同步的手段就是:锁 一般分为公平锁和非公平锁: synchronized(内部锁,互斥锁):synchronized是JVM提供的线程同步机制,如果出现问题,JVM能捕获异常,并释放资源,具体实现机制需要查看JVM源码 synchronized的使用特

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

其实多线程断点下载原理,很简单的,那么我们就来先了解下,如何实现多线程的断点下载,首先:你必须明白第一点,那么就是,什么是多线程下载,该知识点可以查看本博客上一篇文章,Android之多线程下载原理,断点下载呢,其实就是在这个的基础之上添加了一些东西,那么添加了什么东西了,现在来做一个详细的了解. 1.在下载的过程中,边下载,变用一个文件来记录下载的位置,也就是下载了多少的数据 1.创建文件 2.记录下载多少数据 3.存储数据 2.第二次下载的时候,就去读取文件中是否存有数据,读取上次下载的位置

后台多任务多线程断点下载

忘了上图: 多线程断点下载其实不是很难,主要就是三个方面: 1.根据文件的大小和下载线程的数量,确定每个下载线程要下载的分割文件的大小: 2.记录每个下载线程已经下载完成的进度: 3.将每个线程下载的分割的文件合并到一个文件中. 那么怎么将远程的一个文件分割成三部分来下载呢?其实在HTTP协议中,有一个Range字段,用于客户端到服务器端的请求,可通过该字段指定下载文件的某一段大小,及其单位,格式为:Range: bytes x - y,eg:Range: bytes=0-100 下载从第0 -

33、多线程断点下载的实现&amp;界面的更新

1 <?xml version="1.0" encoding="utf-8"?> 2 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 3 android:layout_width="fill_parent" 4 android:layout_height="fill_parent" 5 and

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

原文:http://www.open-open.com/lib/view/open1423214229232.html 其实多线程断点下载原理,很简单的,那么我们就来先了解下,如何实现多线程的断点下载,首先:你必须明白第一点,那么就是,什么是多线程下载,该知识点可以查看本博客上一篇文章,Android之多线程下载原理,断点下载呢,其实就是在这个的基础之上添加了一些东西,那么添加了什么东西了,现在来做一个详细的了解. 1.在下载的过程中,边下载,变用一个文件来记录下载的位置,也就是下载了多少的数据

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

http://www.cnblogs.com/wendingding/p/3947550.html iOS开发网络篇—多线程断点下载 说明:本文介绍多线程断点下载.项目中使用了苹果自带的类,实现了同时开启多条线程下载一个较大的文件.因为实现过程较为复杂,所以下面贴出完整的代码. 实现思路:下载开始,创建一个和要下载的文件大小相同的文件(如果要下载的文件为100M,那么就在沙盒中创建一个100M的文件,然后计算每一段的下载量,开启多条线程下载各段的数据,分别写入对应的文件部分). 项目中用到的主要

android学习笔记--android启动过程之init.rc文件浅析

1.  init.rc文件结构文件位置:init.c  : /system/core/initinit.rc  : /system/core/rootdir 首先init.rc文件是以模块为单位的,每个模块里的内容都是一起执行的,模块分为3种类型:on.service.import.我们可以看下init.rc文件是怎么写的:1.import import /init.usb.rc import /init.${ro.hardware}.rc import /init.trace.rc 上面的内容