Android 多线程断点下载

package com.itheima.mutiledownloader;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.ProtocolException;
import java.net.URL;
import java.util.ArrayList;
import java.util.List;

import android.os.Bundle;
import android.os.Environment;
import android.os.Handler;
import android.os.Message;
import android.renderscript.Program.TextureType;
import android.text.TextUtils;
import android.app.Activity;
import android.view.Menu;
import android.view.View;
import android.view.ViewGroup.LayoutParams;
import android.widget.EditText;
import android.widget.LinearLayout;
import android.widget.ProgressBar;
import android.widget.Toast;

public class MainActivity extends Activity {
        protected static final int DOWNLOAD_ERROR = 1;
        private static final int THREAD_DOWNLOAD_ERROR = 2;
        public static final int DONWLOAD_FINISH = 3;
        private EditText et_path;
        private EditText et_number;
        private LinearLayout ll_pb;
        private String path;
        private static int threadCount = 4;
        private static int blockSize = 0;
        private static int livingThread = 0;

        private List<ProgressBar> pbs;

        private Handler handler = new Handler() {
                public void handleMessage(android.os.Message msg) {
                        switch (msg.what) {
                        case DOWNLOAD_ERROR:
                                Toast.makeText(getApplicationContext(), "初始化下载失败", 1).show();
                                break;
                        case THREAD_DOWNLOAD_ERROR:
                                String str = (String) msg.obj;
                                Toast.makeText(getApplicationContext(), str, 1).show();
                                break;
                        case DONWLOAD_FINISH:
                                Toast.makeText(getApplicationContext(), "下载完毕", 1).show();
                                findViewById(R.id.bt).setEnabled(true);
                                break;
                        }
                };
        };

        @Override
        protected void onCreate(Bundle savedInstanceState) {
                super.onCreate(savedInstanceState);
                setContentView(R.layout.activity_main);
                et_path = (EditText) findViewById(R.id.et_path);
                et_number = (EditText) findViewById(R.id.et_number);
                ll_pb = (LinearLayout) findViewById(R.id.ll_pb);

        }

        public void click(View view) {
                path = et_path.getText().toString().trim();
                String strNumber = et_number.getText().toString().trim();
                if (TextUtils.isEmpty(path) || TextUtils.isEmpty(strNumber)) {
                        Toast.makeText(this, "请输入参数", 0).show();
                        return;
                } else {
                        // 下载的线程的个数
                        threadCount = Integer.parseInt(strNumber);
                        pbs = new ArrayList<ProgressBar>();
                        ll_pb.removeAllViews();
                        for (int i = 0; i < threadCount; i++) {
                                ProgressBar pb = (ProgressBar) View.inflate(
                                                getApplicationContext(), R.layout.pb_item, null);
                                pb.setPadding(5, 5, 5, 5);
                                ll_pb.addView(pb, LayoutParams.MATCH_PARENT, 12);
                                //在集合里面添加进度条的引用.
                                pbs.add(pb);
                        }
                        findViewById(R.id.bt).setEnabled(false);
                        new Thread() {
                                public void run() {
                                        try {
                                                URL url = new URL(path);
                                                HttpURLConnection conn = (HttpURLConnection) url
                                                                .openConnection();
                                                conn.setRequestMethod("GET");
                                                conn.setConnectTimeout(5000);
                                                // 服务器文件的长度
                                                int length = conn.getContentLength();
                                                // 本地创建一个大小跟服务器一样的文件
                                                RandomAccessFile raf = new RandomAccessFile(Environment
                                                                .getExternalStorageDirectory()
                                                                .getAbsolutePath()
                                                                + "/" + getFileName(path), "rw");
                                                raf.setLength(length);
                                                System.out.println("文件的总长度:" + length);
                                                // 计算每个线程下载的位置.
                                                blockSize = length / threadCount;
                                                System.out.println("每一个块的平均长度:" + blockSize);
                                                livingThread = threadCount;
                                                for (int i = 1; i <= threadCount; i++) {
                                                        int startIndex = (i - 1) * blockSize;
                                                        int endIndex = i * blockSize - 1;
                                                        if (i == threadCount) {
                                                                endIndex = length;
                                                        }
                                                        //设置进度条的总进度
                                                        pbs.get(i-1).setMax(endIndex-startIndex);
                                                        new Thread(
                                                                        new DownLoadTask(i, startIndex, endIndex))
                                                                        .start();
                                                }
                                        } catch (Exception e) {
                                                e.printStackTrace();
                                                Message msg = Message.obtain();
                                                msg.what = DOWNLOAD_ERROR;
                                                handler.sendMessage(msg);
                                        }
                                };
                        }.start();

                }
        }

        /**
         * 获取下载的文件的名称
         *
         * @param path
         * @return
         */
        private static String getFileName(String path) {
                int start = path.lastIndexOf("/") + 1;
                return path.substring(start);
        }

        private class DownLoadTask implements Runnable {

                private int id;
                private int startIndex;
                private int endIndex;

                public DownLoadTask(int id, int startIndex, int endIndex) {
                        this.id = id;
                        this.startIndex = startIndex;
                        this.endIndex = endIndex;
                }

                @Override
                public void run() {
                        try {
                                int downloadCount =  0 ;
                                URL url = new URL(path);
                                HttpURLConnection conn = (HttpURLConnection) url
                                                .openConnection();
                                conn.setRequestMethod("GET");
                                conn.setConnectTimeout(5000);
                                // 在每次下载之前,检查是否存在已经下载大小的记录文件.
                                File f = new File(Environment.getExternalStorageDirectory()
                                                .getAbsolutePath()
                                                + "/"
                                                + getFileName(path)
                                                + id
                                                + ".txt");
                                if (f.exists() && f.length() > 0) {
                                        FileInputStream fis = new FileInputStream(f);
                                        BufferedReader br = new BufferedReader(
                                                        new InputStreamReader(fis));
                                        // 已经下载的大小.
                                        downloadCount = Integer.parseInt(br.readLine());
                                        startIndex += downloadCount;
                                        br.close();
                                        fis.close();
                                }
                                conn.setRequestProperty("Range", "bytes=" + startIndex + "-"
                                                + endIndex);
                                System.out.println("线程id:" + id + " 下载的位置:" + startIndex
                                                + "~~~" + endIndex);
                                RandomAccessFile raf = new RandomAccessFile(Environment
                                                .getExternalStorageDirectory().getAbsolutePath()
                                                + "/"
                                                + getFileName(path), "rwd");
                                InputStream is = conn.getInputStream();
                                // 指定线程下载存放文件的开始位置.
                                raf.seek(startIndex);
                                int len = 0;
                                byte[] buffer = new byte[4096];
                                int total = 0;// 记录当前线程下载的大小.
                                while ((len = is.read(buffer)) != -1) {
                                        File file = new File(Environment
                                                        .getExternalStorageDirectory().getAbsolutePath()
                                                        + "/" + getFileName(path) + id + ".txt");
                                        RandomAccessFile positionRaf = new RandomAccessFile(file,
                                                        "rwd");
                                        raf.write(buffer, 0, len);
                                        total += len;
                                        positionRaf.write(String.valueOf(total).getBytes());
                                        positionRaf.close();
                                        pbs.get(id-1).setProgress(total+downloadCount);
                                }
                                raf.close();
                                is.close();
                                System.out.println("线程:" + id + "下载完毕...");
                        } catch (Exception e) {
                                e.printStackTrace();
                                Message msg = Message.obtain();
                                msg.what = THREAD_DOWNLOAD_ERROR;
                                msg.obj = "线程:" + id + "下载失败.";
                                handler.sendMessage(msg);
                        } finally {
                                synchronized (MainActivity.this) {
                                        livingThread--;
                                        if (livingThread <= 0) {
                                                System.out.println("全部下载完毕...");
                                                Message msg = Message.obtain();
                                                msg.what = DONWLOAD_FINISH;
                                                handler.sendMessage(msg);
                                                for (int i = 1; i <= threadCount; i++) {
                                                        File f = new File(Environment
                                                                        .getExternalStorageDirectory()
                                                                        .getAbsolutePath()
                                                                        + "/" + getFileName(path) + i + ".txt");
                                                        System.out.println(f.delete());
                                                }
                                        }
                                }

                        }
                }
        }
}
时间: 2024-12-14 02:12:05

Android 多线程断点下载的相关文章

Android 多线程断点下载源码

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

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

我们平时都用"迅雷"下载软件,当下载到一半的时候突然断网,下次开启的时候能够从上次下载的地方继续下载,而且下载速度很快,那么这是怎么做到的呢! 其实它的“快”其实就是多线程的下载实现的,断点下载的原理是将每次下载的字节数存取下来,保证存取的子节点跟下载的同步,并在用户下次下载的时候自动读取 存储点,并以存储点为开始值继续下载.那么android里面如何实现这么断点的下载呢? 在res的布局文件里面先画一个带有进度条的下载界面 activity_main.xml <LinearLa

Android 多线程断点下载(非原创)

1.服务器的CPU分配给每条线程的时间片相同,服务器带宽平均分配给每条线程,所以客户端开启的线程越多,就能抢占到更多的服务器资源,这里在客户端开启多个线程来从服务器下载资源 2.fragment_main.xml <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" andro

Android多线程断点下载的代码流程解析

Step 1:创建一个用来记录线程下载信息的表 创建数据库表,于是乎我们创建一个数据库的管理器类,继承SQLiteOpenHelper类 重写onCreate()与onUpgrade()方法 DBOpenHelper.java: public class DBOpenHelper extends SQLiteOpenHelper { public DBOpenHelper(Context context) { super(context, "downs.db", null, 1); }

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;">

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

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

Android多线程断点续传下载

这个月接到一个项目,要写一个像360助手一样的对于软件管理的APP:其中,遇到了一个问题:多线程断点下载 这个 ,由于之前没有写过这方面的应用功能.所以,不免要自学了.然后就在各个昂站上收索并整理了一下.跟大家分享一下,也能加深我在这方面的理解. 什么是多线程下载? 多线程下载其实就是迅雷,BT一些下载原理,通过多个线程同时和服务器连接,那么你就可以榨取到较高的带宽了,大致做法是将文件切割成N块,每块交给单独一个线程去下载,各自下载完成后将文件块组合成一个文件,程序上要完成做切割和组装的小算法

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

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