Android快速实现断点续传的方法

本文实例为大家分享了Android快速实现断点续传的具体代码,供大家参考,具体内容如下

1.导入依赖

compile 'com.loopj.android:android-async-http:1.4.9'

2.导入权限

<uses-permission android:name="android.permission.INTERNET"></uses-permission>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

3.布局文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  xmlns:app="http://schemas.android.com/apk/res-auto"
  xmlns:tools="http://schemas.android.com/tools"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:orientation="vertical"
  tools:context="com.five.fashion.duandianxuchuan.MainActivity">
<ProgressBar
  style="@style/Widget.AppCompat.ProgressBar.Horizontal"
  android:layout_width="match_parent"
  android:layout_height="100dp"
  android:id="@+id/pb"
  />
<TextView
  android:layout_width="wrap_content"
  android:layout_height="wrap_content"
  android:id="@+id/tv_info"
  />
  <Button
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="下载"
    android:id="@+id/bt_download"
    />
  <Button
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="暂停"
    android:id="@+id/bt_pause"
    />
</LinearLayout>

4.主要代码

package com.five.fashion.duandianxuchuan;

import android.os.Bundle;
import android.os.Environment;
import android.os.Handler;
import android.os.Message;
import android.os.SystemClock;
import android.support.annotation.Nullable;
import android.support.v7.app.AppCompatActivity;
import android.text.TextUtils;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.ProgressBar;
import android.widget.TextView;

import com.loopj.android.http.HttpGet;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.InputStream;
import java.io.RandomAccessFile;

import cz.msebera.android.httpclient.Header;
import cz.msebera.android.httpclient.HttpResponse;
import cz.msebera.android.httpclient.client.HttpClient;
import cz.msebera.android.httpclient.client.methods.HttpHead;
import cz.msebera.android.httpclient.impl.client.DefaultHttpClient;

public class MainActivity extends AppCompatActivity {
  protected static final String TAG = "OtherActivity";
  //下载线程的数量
  private final static int threadsize = 3;
  protected static final int SET_MAX = 0;
  public static final int UPDATE_VIEW = 1;
  private ProgressBar pb;
  private Button bt_download;
  private Button bt_pause;
  private TextView tv_info;
  //显示进度和更新进度
  private Handler mHandler = new Handler(){
    public void handleMessage(android.os.Message msg) {
      switch (msg.what) {
        case SET_MAX://设置进度条的最大值
          int filelength = msg.arg1;
          pb.setMax(filelength);
          break;
        case UPDATE_VIEW://更新进度条 和 下载的比率
          int len = msg.arg1;//新下载的长度
          pb.setProgress(pb.getProgress()+len);//设置进度条的刻度

          int max = pb.getMax();//获取进度的最大值
          int progress = pb.getProgress();//获取已经下载的数据量
          // 下载:30  总:100
          int result = (progress*100)/max;

          tv_info.setText("下载:"+result+"%");

          break;

        default:
          break;
      }
    };
  };
  String uri = "http://c.hiphotos.baidu.com/image/pic/item/b90e7bec54e736d1e51217c292504fc2d46269f3.jpg";
  @Override
  protected void onCreate(@Nullable Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    //找到控件
    pb = (ProgressBar) findViewById(R.id.pb);
    tv_info = (TextView) findViewById(R.id.tv_info);
    bt_download = (Button) findViewById(R.id.bt_download);
    bt_pause = (Button) findViewById(R.id.bt_pause);
    //数据的回显
    //确定下载的文件
    String name = getFileName(uri);
    File file = new File(Environment.getExternalStorageDirectory(), name);
    if (file.exists()){//文件存在回显
      //获取文件的大小
      int filelength = (int) file.length();
      pb.setMax(filelength);
      try {
        //统计原来所有的下载量
        int count = 0;
        //读取下载记录文件
        for (int threadid = 0; threadid < threadsize; threadid++) {
          //获取原来指定线程的下载记录
          int existDownloadLength = readDownloadInfo(threadid);
          count = count + existDownloadLength;
        }
        //设置进度条的刻度
        pb.setProgress(count);

        //计算比率
        int result = (count * 100) / filelength;
        tv_info.setText("下载:" + result + "%");
      } catch (Exception e) {
        e.printStackTrace();
      }
    }
    bt_download.setOnClickListener(new View.OnClickListener() {
      @Override
      public void onClick(View v) {
        download(v);
      }
    });
    bt_pause.setOnClickListener(new View.OnClickListener() {
      @Override
      public void onClick(View v) {
        pause(v);
      }
    });
  }
  //暂停
  private boolean flag = false;//是否在下载

  public void pause(View v){
    flag = false;
    bt_download.setEnabled(true);
    bt_pause.setEnabled(false);
  }

  //下载
  public void download(View v){
    flag = true; //是在下载
    bt_download.setEnabled(false);//一点击变成不可点击
    bt_pause.setEnabled(true);//一点击变成可点击
    new Thread(){//子线程
      public void run() {
        try {
          //获取服务器上文件的大小
          HttpClient client = new DefaultHttpClient();
          HttpHead request = new HttpHead(uri);
          HttpResponse response = client.execute(request);
          //response 只有响应头 没有响应体
          if(response.getStatusLine().getStatusCode() == 200){
            Header[] headers = response.getHeaders("Content-Length");
            String value = headers[0].getValue();
            //文件大小
            int filelength = Integer.parseInt(value);
            Log.i(TAG, "filelength:"+filelength);

            //设置进度条的最大值
            Message msg_setmax = Message.obtain(mHandler, SET_MAX, filelength, 0);
            msg_setmax.sendToTarget();

            //处理下载记录文件
            for(int threadid=0;threadid<threadsize;threadid++){
              //对应的下载记录文件
              File file = new File(Environment.getExternalStorageDirectory(),threadid+".txt");
              //判断文件是否存在
              if(!file.exists()){
                //创建文件
                file.createNewFile();
              }
            }

            //在sdcard创建和服务器大小一样的文件
            String name = getFileName(uri);
            File file = new File(Environment.getExternalStorageDirectory(),name);
            //随机访问文件
            RandomAccessFile raf = new RandomAccessFile(file, "rwd");
            //设置文件的大小
            raf.setLength(filelength);
            //关闭
            raf.close();

            //计算每条线程的下载量
            int block = (filelength%threadsize == 0)?(filelength/threadsize):(filelength/threadsize+1);
            //开启三条线程执行下载
            for(int threadid=0;threadid<threadsize;threadid++){
              new DownloadThread(threadid, uri, file, block).start();
            }
          }
        } catch (Exception e) {
          e.printStackTrace();
        }
      };
    }.start();
  }

  //线程下载类
  private class DownloadThread extends Thread{
    private int threadid;//线程的id
    private String uri;//下载的地址
    private File file;//下载文件
    private int block;//下载的块
    private int start;
    private int end;

    public DownloadThread(int threadid, String uri, File file, int block) {
      super();
      this.threadid = threadid;
      this.uri = uri;
      this.file = file;
      this.block = block;
      //计算下载的开始位置和结束位置
      start = threadid * block;
      end = (threadid + 1)*block -1;

      try {
        //读取该条线程原来的下载记录
        int existDownloadLength = readDownloadInfo(threadid);

        //修改下载的开始位置  从新下载
        start = start + existDownloadLength;
      } catch (Exception e) {
        e.printStackTrace();
      }
    }

    //下载  状态码:200是普通的下载   206是分段下载    Range:范围
    @Override
    public void run() {
      super.run();
      try {
        RandomAccessFile raf = new RandomAccessFile(file, "rwd");
        //跳转到起始位置
        raf.seek(start);

        //分段下载
        HttpClient client = new DefaultHttpClient();
        HttpGet request = new HttpGet(uri);
        request.addHeader("Range", "bytes:"+start+"-"+end);//添加请求头
        HttpResponse response = client.execute(request);
        if(response.getStatusLine().getStatusCode() == 200){
          InputStream inputStream = response.getEntity().getContent();
          //把流写入到文件
          byte[] buffer = new byte[1024];
          int len = 0;
          while((len = inputStream.read(buffer)) != -1){
            //如果暂停下载 点击暂停 false 就直接return 点击下载true接着下载
            if(!flag){
              return;//标准线程结束
            }
            //写数据
            raf.write(buffer, 0, len);

            //读取原来下载的数据量 这里读取是为了更新下载记录
            int existDownloadLength = readDownloadInfo(threadid);//原来下载的数据量

            //计算最新的下载
            int newDownloadLength = existDownloadLength + len;

            //更新下载记录 从新记录最新下载位置
            updateDownloadInfo(threadid, newDownloadLength);
            //更新进度条的显示  下载的百分比
            Message update_msg = Message.obtain(mHandler, UPDATE_VIEW, len, 0);
            update_msg.sendToTarget();
            //模拟 看到进度条动的效果
            SystemClock.sleep(50);
          }
          inputStream.close();
          raf.close();
          Log.i(TAG, "第"+threadid+"条线程下载完成");
        }

      } catch (Exception e) {
        e.printStackTrace();
      }
    }
  }

  /**
   * 读取指定线程的下载数据量
   * @param threadid 线程的id
   * @return
   * @throws Exception
   */
  public int readDownloadInfo(int threadid) throws Exception{
    //下载记录文件
    File file = new File(Environment.getExternalStorageDirectory(),threadid+".txt");
    BufferedReader br = new BufferedReader(new FileReader(file));
    //读取一行数据
    String content = br.readLine();

    int downlength = 0;
    //如果该文件第一次创建去执行读取操作 文件里面的内容是 null
    if(!TextUtils.isEmpty(content)){
      downlength = Integer.parseInt(content);
    }
    //关闭流
    br.close();
    return downlength;
  }

  /**
   * 更新下载记录
   * @param threadid
   * @param newDownloadLength
   */
  public void updateDownloadInfo(int threadid,int newDownloadLength) throws Exception{
    //下载记录文件
    File file = new File(Environment.getExternalStorageDirectory(),threadid+".txt");
    FileWriter fw = new FileWriter(file);
    fw.write(newDownloadLength+"");
    fw.close();
  }

  /**
   * 获取文件的名称
   * @param uri
   * @return
   */
  private String getFileName(String uri){
    return uri.substring(uri.lastIndexOf("/")+1);
  }

}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • android使用OkHttp实现下载的进度监听和断点续传

    1. 导入依赖包 // retrofit, 基于Okhttp,考虑到项目中经常会用到retrofit,就导入这个了. compile 'com.squareup.retrofit2:retrofit:2.1.0' // ButterKnife compile 'com.jakewharton:butterknife:7.0.1' // rxjava 本例中线程切换要用到,代替handler compile 'io.reactivex:rxjava:1.1.6' compile 'io.react

  • Android编程开发实现多线程断点续传下载器实例

    本文实例讲述了Android编程开发实现多线程断点续传下载器.分享给大家供大家参考,具体如下: 使用多线程断点续传下载器在下载的时候多个线程并发可以占用服务器端更多资源,从而加快下载速度,在下载过程中记录每个线程已拷贝数据的数量,如果下载中断,比如无信号断线.电量不足等情况下,这就需要使用到断点续传功能,下次启动时从记录位置继续下载,可避免重复部分的下载.这里采用数据库来记录下载的进度. 效果图:   断点续传 1.断点续传需要在下载过程中记录每条线程的下载进度 2.每次下载开始之前先读取数据库

  • Android多线程+单线程+断点续传+进度条显示下载功能

    效果图 白话分析: 多线程:肯定是多个线程咯 断点:线程停止下载的位置 续传:线程从停止下载的位置上继续下载,直到完成任务为止. 核心分析: 断点: 当前线程已经下载的数据长度 续传: 向服务器请求上次线程停止下载位置的数据 con.setRequestProperty("Range", "bytes=" + start + "-" + end); 分配线程: int currentPartSize = fileSize / mThreadNum

  • Android多线程断点续传下载功能实现代码

    原理 其实断点续传的原理很简单,从字面上理解,所谓断点续传就是从停止的地方重新下载. 断点:线程停止的位置. 续传:从停止的位置重新下载. 用代码解析就是: 断点:当前线程已经下载完成的数据长度. 续传:向服务器请求上次线程停止位置之后的数据. 原理知道了,功能实现起来也简单.每当线程停止时就把已下载的数据长度写入记录文件,当重新下载时,从记录文件读取已经下载了的长度.而这个长度就是所需要的断点. 续传的实现也简单,可以通过设置网络请求参数,请求服务器从指定的位置开始读取数据. 而要实现这两个功

  • Android 断点续传的原理剖析与实例讲解

    本文所要讲的是Android断点续传的内容,以实例的形式进行了详细介绍.   一.断点续传的原理 其实断点续传的原理很简单,就是在http的请求上和一般的下载有所不同而已. 打个比方,浏览器请求服务器上的一个文时,所发出的请求如下: 假设服务器域名为www.jizhuomi.com/android,文件名为down.zip. get /down.zip http/1.1 accept: image/gif, image/x-xbitmap, image/jpeg, image/pjpeg, ap

  • 详解Android使用OKHttp3实现下载(断点续传、显示进度)

    OKHttp3是如今非常流行的Android网络请求框架,那么如何利用Android实现断点续传呢,今天写了个Demo尝试了一下,感觉还是有点意思 准备阶段 我们会用到OKHttp3来做网络请求,使用RxJava来实现线程的切换,并且开启Java8来启用Lambda表达式,毕竟RxJava实现线程切换非常方便,而且数据流的形式也非常舒服,同时Lambda和RxJava配合食用味道更佳 打开我们的app Module下的build.gradle,代码如下 apply plugin: 'com.an

  • Android FTP 多线程断点续传下载\上传的实例

    最近在给我的开源下载框架Aria增加FTP断点续传下载和上传功能,在此过程中,爬了FTP的不少坑,终于将功能实现了,在此把一些核心功能点记录下载. FTP下载原理 FTP单线程断点续传 FTP和传统的HTTP协议有所不同,由于FTP没有所谓的头文件,因此我们不能像HTTP那样通过设置header向服务器指定下载区间. 但是FTP协议提供了一个更好用的命令REST用于从指定位置恢复任务,同时FTP协议也提供了一个命令SIZE用于获取下载的文件大小,有了这两个命令,FTP断点续传也就没有什么问题.

  • Android实现网络多线程断点续传下载实例

    我们编写的是Andorid的HTTP协议多线程断点下载应用程序.直接使用单线程下载HTTP文件对我们来说是一件非常简单的事.那么,多线程断点需要什么功能? 1.多线程下载, 2.支持断点. 使用多线程的好处:使用多线程下载会提升文件下载的速度.那么多线程下载文件的过程是: (1)首先获得下载文件的长度,然后设置本地文件的长度. HttpURLConnection.getContentLength();//获取下载文件的长度 RandomAccessFile file = new RandomAc

  • android实现多线程下载文件(支持暂停、取消、断点续传)

    多线程下载文件(支持暂停.取消.断点续传) 多线程同时下载文件即:在同一时间内通过多个线程对同一个请求地址发起多个请求,将需要下载的数据分割成多个部分,同时下载,每个线程只负责下载其中的一部分,最后将每一个线程下载的部分组装起来即可. 涉及的知识及问题 请求的数据如何分段 分段完成后如何下载和下载完成后如何组装到一起 暂停下载和继续下载的实现(wait().notifyAll().synchronized的使用) 取消下载和断点续传的实现 一.请求的数据如何分段 首先通过HttpURLConne

  • Android 断点续传原理以及实现

    Android 断点续传原理以及实现 0.  前言 在Android开发中,断点续传听起来挺容易,在下载一个文件时点击暂停任务暂停,点击开始会继续下载文件.但是真正实现起来知识点还是蛮多的,因此今天有时间实现了一下,并进行记录. 1.  断点续传原理 在本地下载过程中要使用数据库实时存储到底存储到文件的哪个位置了,这样点击开始继续传递时,才能通过HTTP的GET请求中的setRequestProperty()方法可以告诉服务器,数据从哪里开始,到哪里结束.同时在本地的文件写入时,RandomAc

随机推荐