Android 断点续传解析

2024-05-11 00:38
文章标签 android 解析 断点续传

本文主要是介绍Android 断点续传解析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、原理


断点续传,顾名思义就是下载文件时不必每次都重新开始,可以从之前已经下载好的地方接着下载,这样既可以省流量还能省时间。那么怎么样才能做到呢?这就要靠 RandomAccessFile 这个类了,来看看 RandomAccessFile 这个类的定义:

public class RandomAccessFile implements DataInput, DataOutput, Closeable {.......
}

下面来看一个简单使用 RandomAccessFile 的 demo:

public class RandomIoDemo {private static int len;public static void main(String[] args) throws Exception {// 在磁盘中预先创建一个文件,分配预定的空间RandomAccessFile raf = new RandomAccessFile("result.txt", "rwd");raf.setLength(1024); // 预分配 1kb 的文件空间raf.close();// 所要写入的文件内容String s1 = "第一个字符串的内容";String s2 = "第二个字符串的内容";String s3 = "第三个字符串的内容";String s4 = "第四个字符串的内容";String s5 = "第五个字符串的内容";len = s1.getBytes().length;// 利用多线程同时写入一个文件new FileWriteThread(0, s1.getBytes()).start();new FileWriteThread(len, s2.getBytes()).start();new FileWriteThread(len * 2, s3.getBytes()).start();new FileWriteThread(len * 3, s4.getBytes()).start();new FileWriteThread(len * 4, s5.getBytes()).start();}// 利用线程在文件的指定位置写入指定数据private static class FileWriteThread extends Thread {private int skip;private byte[] content;/*** @param skip 写入文件需要跳过的字节数* @param content 写入到文件的内容*/private FileWriteThread(int skip, byte[] content) {this.skip = skip;this.content = content;}public void run() {try {FileChannel channel = new RandomAccessFile("result.txt", "rwd").getChannel();MappedByteBuffer buffer = channel.map(FileChannel.MapMode.READ_WRITE, skip, len);buffer.put(content);} catch (IOException e) {// TODO Auto-generated catch blocke.printStackTrace();}}}}

这个一个简单的 Java 实现,功能很简单,就是把 s1~s5 这几个字符串的内容写入到 result.txt 这个文本文件中去。为了方便起见这几个 s1~s5 这几个字符串的大小都是相同的;你可能会说这样一个功能很简单呀,用 StringBuffer 就可以实现。但是如果s1~s5 这几个字符串的长度很长,或者说要写入到最终文件的内容不是字符串,而是音频、图片流之类的,那么使用 RandomAccessFile 就可以展现出他的优势了。概括来说,RandomAccessFile 可以实现文件从特定的位置进行读写。

 

二、基于OKHttp的断点下载简单实现

RandomAccessFile 只是提供了一种文件类型,方便我们进行断点续传,那么如果要实现断点下载的功能,我们需要思考以下两个问题。

  • 首先,怎么判断一个服务器上的文件是否支持断点下载?
  • 其次,如果这个文件支持断点下载,那么怎么告知服务器端我要从哪个字节开始下载?

这两个疑问可以通过下面的代码得到答案:

public class DownloadHelper {public static OkHttpClient mClient = new OkHttpClient();private static Call mCall;public static void startDownload(int startPoint, int endPoint, Handler mHandler) {Request request = new Request.Builder().url(Constants.PACKAGE_URL).header("RANGE", "bytes=" + startPoint + "-" + endPoint).build();mCall = mClient.newCall(request);mCall.enqueue(new OkHttpCallback(startPoint, mHandler));}public static void startDownload(int startPoint, Handler mHandler) {Request request = new Request.Builder().url(Constants.PACKAGE_URL).header("RANGE", "bytes=" + startPoint + "-").build();mCall = mClient.newCall(request);mCall.enqueue(new OkHttpCallback(startPoint, mHandler));}public static void cancelDownload() {if (mCall != null) {mCall.cancel();}}
}

可以看到,通过设置 Request 对象的 header 方法的 RANGE 就可以告知服务器端开始下载的节点。我们再看 OkHttpCallback 的实现:

public class OkHttpCallback implements Callback {private Handler mHandler;private int startPoint;public OkHttpCallback(int startPoint, Handler mHandler) {this.startPoint = startPoint;this.mHandler = mHandler;}@Overridepublic void onFailure(Call call, IOException e) {mHandler.sendEmptyMessage(100);}@Overridepublic void onResponse(Call call, Response response) {if (response.code() != HttpURLConnection.HTTP_PARTIAL) {//返回code非206 ,不支持断点续传mHandler.sendEmptyMessage(400);return;}FileChannel fileChannel = null;ResponseBody body = response.body();int total = (int) body.contentLength();int currentLength = 0;InputStream inputStream = body.byteStream();try {RandomAccessFile randomAccessFile = new RandomAccessFile(Constants.FILE_PATH, "rws");fileChannel = randomAccessFile.getChannel();Log.e(TAG, "onResponse: startPoint=" + startPoint + " ,total=" + total);MappedByteBuffer mappedByteBuffer = fileChannel.map(FileChannel.MapMode.READ_WRITE, startPoint, total);int len;byte[] buffer = new byte[1024];while ((len = inputStream.read(buffer)) != -1) {currentLength = currentLength + len;mappedByteBuffer.put(buffer, 0, len);Message msg = Message.obtain();msg.arg1 = total;msg.arg2 = currentLength;msg.what = 300;mHandler.sendMessage(msg);}} catch (IOException e) {e.printStackTrace();} finally {try {inputStream.close();if (fileChannel != null) {fileChannel.close();}} catch (IOException e) {e.printStackTrace();}}}
}

在 onResponse 回调方法中我们可以看到,当我们在之前的 head 中添加了 RANGE 字段,但是如果返回的 http code 不是 206,我们就可以确定所请求的文件是不支持断点下载的。现在就可以非常方便的实现一个简单的断点续传功能了。

class MyHandler extends Handler {@Overridepublic void handleMessage(Message msg) {super.handleMessage(msg);switch (msg.what) {case 400:Toast.makeText(mContext, "不支持断点续传", Toast.LENGTH_SHORT).show();break;case 100:Toast.makeText(mContext, "fail", Toast.LENGTH_SHORT).show();break;case 300:int total = msg.arg1;int current = msg.arg2;if (!isPause && !isStop) {totalValue = current + breakPointValue;int percent = (int) (totalValue * 100f / (total + breakPointValue));if (percent < 100) {mProgressBar.setProgress(percent);progressValue.setText(String.valueOf(percent));} else {Intent intent = new Intent(Intent.ACTION_VIEW);intent.setDataAndType(Uri.parse("file://" + Constants.FILE_PATH),"application/vnd.android.package-archive");mContext.startActivity(intent);resetStatus();}}break;default:break;}}}isPause=true;pause.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {if (isPause) {pause.setImageResource(R.drawable.ic_pause_circle_outline_black_24dp);DownloadHelper.startDownload(breakPointValue, mMyHandler);} else {pause.setImageResource(R.drawable.ic_play_circle_outline_black_24dp);DownloadHelper.cancelDownload();breakPointValue = totalValue;}isPause = !isPause;}});

breakPointValue 这个变量记录了每次暂停下载时断点位置已完成的下载量,第一次开始下载时他的初始值为 0,因此便开始从头下载这个文件,并通过 Handler 依次累加已经完成的下载量 totalValue,同时更新下载进度;当暂停时停止下载任务,breakPointValue 的值就是此刻的总下载量,再次点击继续下载,此时 breakPointValue 就会从上次断掉的位置开始新一次的下载任务;依次类推直到下载完成。这样,就简单的完成了一个文件的断点下载任务。

这里再总结一下需要注意的地方:

  • 使用 APK 类型的文件作为断点下载的测试非常有针对性,如果断点续传的过程中数据错误或丢失,将导致最终下载的完成的 APK 文件破损而无法安装。
  • 在 Http 的 ResponseBody 中,contentLength 的值不是一成不变的,他每次返回的值并不是当前所请求文件实际的大小,而是此次请求能够传输的大小,也就是从文件总大小 - RANGE 所包含的大小。因此,需要每次把上一次暂停时 breakPointValue 的值作为下一次累加值的基数。

这篇关于Android 断点续传解析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/978022

相关文章

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

一文解析C#中的StringSplitOptions枚举

《一文解析C#中的StringSplitOptions枚举》StringSplitOptions是C#中的一个枚举类型,用于控制string.Split()方法分割字符串时的行为,核心作用是处理分割后... 目录C#的StringSplitOptions枚举1.StringSplitOptions枚举的常用

Python函数作用域与闭包举例深度解析

《Python函数作用域与闭包举例深度解析》Python函数的作用域规则和闭包是编程中的关键概念,它们决定了变量的访问和生命周期,:本文主要介绍Python函数作用域与闭包的相关资料,文中通过代码... 目录1. 基础作用域访问示例1:访问全局变量示例2:访问外层函数变量2. 闭包基础示例3:简单闭包示例4

MyBatis延迟加载与多级缓存全解析

《MyBatis延迟加载与多级缓存全解析》文章介绍MyBatis的延迟加载与多级缓存机制,延迟加载按需加载关联数据提升性能,一级缓存会话级默认开启,二级缓存工厂级支持跨会话共享,增删改操作会清空对应缓... 目录MyBATis延迟加载策略一对多示例一对多示例MyBatis框架的缓存一级缓存二级缓存MyBat

前端缓存策略的自解方案全解析

《前端缓存策略的自解方案全解析》缓存从来都是前端的一个痛点,很多前端搞不清楚缓存到底是何物,:本文主要介绍前端缓存的自解方案,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录一、为什么“清缓存”成了技术圈的梗二、先给缓存“把个脉”:浏览器到底缓存了谁?三、设计思路:把“发版”做成“自愈”四、代码

Java集合之Iterator迭代器实现代码解析

《Java集合之Iterator迭代器实现代码解析》迭代器Iterator是Java集合框架中的一个核心接口,位于java.util包下,它定义了一种标准的元素访问机制,为各种集合类型提供了一种统一的... 目录一、什么是Iterator二、Iterator的核心方法三、基本使用示例四、Iterator的工

Java JDK Validation 注解解析与使用方法验证

《JavaJDKValidation注解解析与使用方法验证》JakartaValidation提供了一种声明式、标准化的方式来验证Java对象,与框架无关,可以方便地集成到各种Java应用中,... 目录核心概念1. 主要注解基本约束注解其他常用注解2. 核心接口使用方法1. 基本使用添加依赖 (Maven

Java中的分布式系统开发基于 Zookeeper 与 Dubbo 的应用案例解析

《Java中的分布式系统开发基于Zookeeper与Dubbo的应用案例解析》本文将通过实际案例,带你走进基于Zookeeper与Dubbo的分布式系统开发,本文通过实例代码给大家介绍的非常详... 目录Java 中的分布式系统开发基于 Zookeeper 与 Dubbo 的应用案例一、分布式系统中的挑战二