Android 断点续传进阶之多线程下载
今天继续下载的风骚走位内容—多线程多文件断点续传
Android 断点续传基础之单线程下载:http://blog.csdn.net/qq_27489007/article/details/53897653
效果图:
文件关系:
所需内容
多文件下载列表的显示
启动多个线程分段下载
使用通知栏显示进度条
使用其他方式实现线程通信
与单线程不主要不同:
任务下载类(DuoDownloadTask) 使用了线程池, 数据库对线程的增、删操作使用synchronized
多线程下载原理简介
假设要分3个线程下载一个100字节的文件,每个线程可以平分线程1(0-32字节)、 线程2(33-65字节)、线程3(66-100字节)
部分主要代码:
/**
-
多线程多文件的下载任务类,为了学习查看方便 所以提出来了
-
Created by lung on 2016-12-17.
*/
public class DuoDownloadTask {
private Context mContext = null;
private FileInfo mFileInfo = null;
private ThreadDAOImpl mThreadDAO = null;
private long mFinished = 0; //总的完成进度
public boolean isPause = false; //暂停下载的开关
private int mThreadCount = 1; //线程数量
private List downloadThreadList = null;//线程集合 方便管理
public static ExecutorService executorService = Executors.newCachedThreadPool();//线程池
public DuoDownloadTask(Context mContext, FileInfo mFileInfo, int mThreadCount) {
this.mContext = mContext;
this.mFileInfo = mFileInfo;
this.mThreadCount = mThreadCount;
mThreadDAO = new ThreadDAOImpl(mContext);
}
//下载的方法
public void download() {
//读取数据库的线程信息
List threaddInfos = mThreadDAO.getThreads(mFileInfo.getUrl());
if (threaddInfos.size() == 0) { //如果数据库无线程信息
//获得每个线程下载的长度
int length = mFileInfo.getLength() / mThreadCount;
for (int i = 0; i < mThreadCount; i++) {
// 参数三线程开始的地方 参数四 线程结束的地方 参数五任务完成的进度
TheardInfo theardInfo = new TheardInfo(i, mFileInfo.getUrl(), length * i, (i + 1) * length - 1, 0);
//最后一个线程 会出现除不尽的情况
if (i == mThreadCount - 1) {
//直接把线程的结束位置 设置成文件的最大长度位置
theardInfo.setEnd(mFileInfo.getLength());
}
//添加到线程信息集合中
threaddInfos.add(theardInfo);
//向数据库插入线程信息
mThreadDAO.insertThread(theardInfo);
}
}
downloadThreadList = new ArrayList<>();
//启动多个线程进行下载
for (TheardInfo info : threaddInfos) {
DownloadThread thread = new DownloadThread(info);
// thread.start(); //开始线程添加到集合中
DuoDownloadTask.executorService.execute(thread); //利用线程池来执行线程任务
downloadThreadList.add(thread); //把线程添加进去方便ckeckAllThreadFinshed检查线程
}
}
//这个方法 用来判断所有的线程是否都执行完了
private synchronized void checkAllThreadFinshed() {
boolean allFinished = true;
//遍历线程集合,判断线程是否都执行完毕
for (DownloadThread thread : downloadThreadList) {
if (!thread.isFinished) {
allFinished = false;
break;
}
}
if (allFinished) { //如果线程都完成了
//下载完成后 删除线程信息
//删除下载记录
mThreadDAO.deleteThread(mFileInfo.getUrl(), mFileInfo.getId());
//发送广播通知ui下载任务结束
Intent intent = new Intent(DuoDownloadService.ACTION_FINISHED);
intent.putExtra(“fileInfo”, mFileInfo);
mContext.sendBroadcast(intent); //发送广播
}
}
class DownloadThread extends Thread {
private TheardInfo threadInfo;
public boolean isFinished = false; //用来标识线程是否结束
public DownloadThread(TheardInfo threadInfo) {
this.threadInfo = threadInfo;
}
@Override
public void run() {
//设置下载位置
try {
URL url = new URL(threadInfo.getUrl());
HttpURLConnection conn = (HttpURLConnection) url.openConnection();
conn.setRequestMethod(“GET”);
conn.setConnectTimeout(3000);
//开始的字节数 为开始加上完成的长度
long start = threadInfo.getStart() + threadInfo.getFinished();
//下载的范围 开始的字节数 到结束的字节数
conn.setRequestProperty(“Range”, “bytes=” + start + “-” + threadInfo.getEnd());
//设置文件写入位置 路径 文件名
File file = new File(DownloadService.DOWNLOAD_PATH, mFileInfo.getFileName());
//
RandomAccessFile raf = new RandomAccessFile(file, “rwd”);
raf.seek(start); //文件的写入位置
Intent intent = new Intent(DownloadService.ACTION_UPDATE);//把进度广播发送给activity 所以需要intent
mFinished += threadInfo.getFinished();//从线程中拿到完成的进度
//开始下载
if (conn.getResponseCode() == 206) {
//读取数据
InputStream input = conn.getInputStream();
byte[] buffer = new byte[1024 * 4];
int len = -1;
long time = System.currentTimeMillis(); //拿到当前时间
while ((len = input.read(buffer)) != -1) {
//写入文件
raf.write(buffer, 0, len);
//把下载进度发送广播给activity
mFinished += len; //把现在下载的进度累加进去
//累加每个线程完成的进度
threadInfo.setFinished(threadInfo.getFinished() + len);
if (System.currentTimeMillis() - time > 1000) { //减少ni负载 大于10秒发送更新
time = System.currentTimeMillis();
//以百分比的形式发送给广播
intent.putExtra(“finished”, (int) (mFinished * 100 / mFileInfo.getLengt
h()));
intent.putExtra(“id”, mFileInfo.getId());
mContext.sendBroadcast(intent);
}
//在下载暂停时,保存下载进度
if (isPause) { //如果暂停 把线程信息进行保存 最后一个参数 他会把每个线程的 进度保存起来
mThreadDAO.updateThread(mFileInfo.getUrl(), mFileInfo.getId(), threadInfo.getFinished());
return;
}
}
intent.putExtra(“finished”, 100);
mContext.sendBroadcast(intent); //发送广播
// 标识线程执行完毕
isFinished = true;
//每个线程执行完毕 都执行去判断 所有的线程是否都执行完毕
checkAllThreadFinshed();
input.close();
}
raf.close();
conn.disconnect();
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally { //关闭各种链接
}
super.run();
}
}
}
ThreadDAOImpl.java
相关文章:
Android 断点续传进阶之多线程下载
今天继续下载的风骚走位内容—多线程多文件断点续传 Android 断点续传基础之单线程下载:http://blog.csdn.net/qq_27489007/article/details/53897653 效果图: 文件关系: 所需内容 多文件下载列表的显示 启动多个线程分段下载 使用通知栏…...
Python爬虫学习 | Scrapy框架详解
一.Scrapy框架简介 何为框架,就相当于一个封装了很多功能的结构体,它帮我们把主要的结构给搭建好了,我们只需往骨架里添加内容就行。scrapy框架是一个为了爬取网站数据,提取数据的框架,我们熟知爬虫总共有四大部分&am…...
用户态协议栈05—架构优化
优化部分 添加了in和out两个环形缓冲区,收到数据包后添加到in队列;经过消费者线程处理之后,将需要发送的数据包添加到out队列。添加数据包解析线程(消费者线程),架构分层 #include <rte_eal.h> #inc…...
模拟退火算法
模拟退火算法(Simulated Annealing, SA)是一种用于全局优化问题的概率搜索算法,其灵感来自于金属退火过程。在金属退火中,材料被加热到高温,然后缓慢冷却,以减少其晶格中的缺陷并达到最小能量状态。模拟退火…...
Java匿名类
Java 匿名类是一种特殊的内部类,它没有名字,并且通常用来简化代码实现,尤其是在实现接口或者抽象类的实例时。匿名类可以在实例化时定义其行为,而不需要创建单独的类文件。 匿名类的特点 没有名字:匿名类是没有名字的…...
G7易流赋能化工物流,实现安全、环保与效率的共赢
近日,中国物流与采购联合会在古都西安举办了备受瞩目的第七届化工物流安全环保发展论坛。以"坚守安全底线,追求绿色发展,智能规划化工物流未来"为主题,该论坛吸引了众多政府部门、行业专家和企业代表的参与。G7易流作为…...
y=sin(2x)
函数 \( y \sin(2x) \) 是一个正弦函数,其中 \( x \) 是自变量,\( y \) 是因变量。这个函数描述了一个周期性波动的波形,其特点是: 1. **振幅**:正弦函数的振幅是 1,这意味着波形在 \( y \) 轴上的最大值…...
快捷方式(lnk)--加载HTA-CS上线
免责声明:本文仅做技术交流与学习... 目录 CS: HTA文档 文件托管 借助mshta.exe突破 本地生成lnk快捷方式: 非系统图标路径不同问题: 关于lnk的上线问题: CS: HTA文档 配置监听器 有效载荷---->HTA文档--->选择监听器--->选择powershell模式----> 默认生成一…...
从同—视角理解扩散模型(Understanding Diffusion Models A Unified Perspective)
从同—视角理解扩散模型 Understanding Diffusion Models A Unified Perspective【全公式推导】【免费视频讲解】 B站视频讲解 视频的论文笔记 从同一视角理解扩散模型【视频讲解笔记】 配合视频讲解的同步笔记。 整个系列完整的论文笔记内容如下,仅为了不用—一回复…...
docker 基本用法及跨平台使用
一、Docker的优点 docker 主要解决的问题就是程序开发过程中编译和部署中遇到的环境配置的问题。 1.1 Docker与其他虚拟机层次结构的区别** 运行程序重点关注点在于环境。 VM虚拟机是基于Hypervisor虚拟化服务运行的。 Docker是基于内核的虚拟化技术实现的。 1.2 Docker的技…...
Vscode远程ubuntu
远程连接 到这里vscode远程到ubuntu和关闭远程连接,已完成 配置python环境 在远程目录下新建.vscode隐藏文件夹,文件夹里新建一个 settings.json 文件, 先远程服务器看下conda下的python虚拟环境位置 settings.json位置及内容如下 测试pyt…...
SHA256 安全散列算法加速器实验
1、SHA256 介绍 SHA256 加速器是用来计算 SHA-256 的计算单元, SHA256 是 SHA-2 下细分出的一种算法。 SHA-2 名称来自于安全散列算法 2 (英语: Secure Hash Algorithm 2 )的缩写,一种密码散列函 数算法标准…...
Elasticsearch-ES查询单字段去重
ES 语句 整体数据 GET wkl_test/_search {"query": {"match_all": {}} }结果: {"took" : 123,"timed_out" : false,"_shards" : {"total" : 1,"successful" : 1,"skipped" : 0…...
【Apache Doris】周FAQ集锦:第 7 期
【Apache Doris】周FAQ集锦:第 7 期 SQL问题数据操作问题运维常见问题其它问题关于社区 欢迎查阅本周的 Apache Doris 社区 FAQ 栏目! 在这个栏目中,每周将筛选社区反馈的热门问题和话题,重点回答并进行深入探讨。旨在为广大用户和…...
EE trade:炒伦敦金的注意事项及交易指南
在贵金属市场中,伦敦金因其高流动性和全球认可度,成为广大投资者的首选。然而,在炒伦敦金的过程中,投资者需要注意一些关键点。南华金业小编带您一起来看看。 国际黄金报价 一般国际黄金报价会提供三个价格: 买价(B…...
JAVA医院绩效考核系统源码 功能特点:大型医院绩效考核系统源码
JAVA医院绩效考核系统源码 功能特点:大型医院绩效考核系统源码 医院绩效管理系统主要用于对科室和岗位的工作量、工作质量、服务质量进行全面考核,并对科室绩效工资和岗位绩效工资进行核算的系统。医院绩效管理系统开发主要用到的管理工具有RBRVS、DRGS…...
Python神经影像数据的处理和分析库之nipy使用详解
概要 神经影像学(Neuroimaging)是神经科学中一个重要的分支,主要研究通过影像技术获取和分析大脑结构和功能的信息。nipy(Neuroimaging in Python)是一个强大的 Python 库,专门用于神经影像数据的处理和分析。nipy 提供了一系列工具和方法,帮助研究人员高效地处理神经影…...
非关系型数据库NoSQL数据层解决方案 之 Mongodb 简介 下载安装 springboot整合与读写操作
MongoDB 简介 MongoDB是一个开源的面向文档的NoSQL数据库,它采用了分布式文件存储的数据结构,是当前非常流行的数据库之一。 以下是MongoDB的主要特点和优势: 面向文档的存储: MongoDB是一个面向文档的数据库管理系统࿰…...
使用Redis优化Java应用的性能
使用Redis优化Java应用的性能 大家好,我是免费搭建查券返利机器人省钱赚佣金就用微赚淘客系统3.0的小编,也是冬天不穿秋裤,天冷也要风度的程序猿!今天我们来探讨如何使用Redis优化Java应用的性能。Redis是一种开源的内存数据结构…...
基于Python的数据可视化大屏的设计与实现
基于Python的数据可视化大屏的设计与实现 Design and Implementation of Python-based Data Visualization Dashboard 完整下载链接:基于Python的数据可视化大屏的设计与实现 文章目录 基于Python的数据可视化大屏的设计与实现摘要第一章 导论1.1 研究背景1.2 研究目的1.3 研…...
什么是N卡和A卡?有什么区别?
名人说:莫听穿林打叶声,何妨吟啸且徐行。—— 苏轼《定风波莫听穿林打叶声》 本篇笔记整理:Code_流苏(CSDN)(一个喜欢古诗词和编程的Coder😊) 目录 一、什么是N卡和A卡?有什么区别?…...
四边形不等式优化
四边形不等式优化 应用于类似以下dp转移方程。 f i min 1 ≤ j ≤ i ( w i , j , f i ) f_{i}\min_{1\le j\le i}(w_{i,j},f_{i}) fi1≤j≤imin(wi,j,fi) 假设 w i , j w_{i,j} wi,j 可以在 O ( 1 ) O(1) O(1) 的时间内进行计算。 在正常情况下,…...
这家民营银行起诉担保公司?暴露担保增信兜底隐患
来源 | 镭射财经(leishecaijing) 助贷领域中,各路资方依赖担保增信业务扩张数年,其风险积压也不容忽视。一旦助贷平台或担保公司兜不住底,资方就将陷入被动。 最近,一则民营银行起诉合作担保公司的消息引…...
vscode禅模式怎么退出
1、如何进入禅模式:查看--外观--禅模式 2、退出禅模式 按二次ESC,就可以退出。...
Java23种设计模式(四)
1、备忘录模式 备忘录模式(Memento Pattern)保存一个对象的某个状态,以便在适当的时候恢复对象,备忘录模式属于行为型模式。 备忘录模式允许在不破坏封装性的前提下,捕获和恢复对象的内部状态。 实现方式 创建备忘录…...
HTML静态网页成品作业(HTML+CSS)——故宫介绍网页(4个页面)
🎉不定期分享源码,关注不丢失哦 文章目录 一、作品介绍二、作品演示三、代码目录四、网站代码HTML部分代码 五、源码获取 一、作品介绍 🏷️本套采用HTMLCSS,未使用Javacsript代码,共有4个页面。 二、作品演示 三、代…...
Zookeeper:客户端命令行操作
文章目录 一、help二、ls path三、create四、get path五、set六、stat七、delete八、deleteall 一、help 显示所有操作命令。 二、ls path 使用ls命令来查看当前znode的子节点[可监听] w:监听子节点变化。s:附加次级信息。 三、create 普通创建&am…...
区块链技术介绍和用法
区块链技术是一种分布式账本技术,可以记录和存储一系列交易信息,并通过密码学算法保证信息的安全性和不可篡改性。区块链技术的核心概念是“区块”和“链”。 每个区块包含了一部分交易信息,以及一个指向上一个区块的哈希值。当新的交易发生…...
Upload-Labs-Linux1 使用 一句话木马
解题步骤: 1.新建一个php文件,编写内容: <?php eval($_REQUEST[123]) ?> 2.将编写好的php文件上传,但是发现被阻止,网站只能上传图片文件。 3.解决方法: 将php文件改为图片文件(例…...
从 Hadoop 迁移,无需淘汰和替换
我们仍然惊讶于有如此多的客户来找我们,希望从HDFS迁移到现代对象存储,如MinIO。我们现在以为每个人都已经完成了过渡,但每周,我们都会与一个决定进行过渡的主要、高技术性组织交谈。 很多时候,在这些讨论中ÿ…...
怎么查网站接入商/seo顾问阿亮
在vue中使用dom-top-image,可截取有滚动条的页面,支持多种格式 具体方法如下: 1.首选安装相关插件 npm install dom-to-image然后在使用的页面中引用 import domtoimage from dom-to-image下面是封装的具体的方法: shotPic () {const that thisconst …...
网站怎么做排名/百度推广怎么操作流程
什么是过滤器?有什么用?过滤器JavaWeb三大组件之一,它与Servlet很相似。不过滤器是用来拦截请求的,而不是处理请求的。过滤,顾名思义,就是留下我们想要的,丢掉我们不需要的。例如:某…...
怎么用二维动画做网站首页步骤/爱站网怎么用
在MySQL服务器出现短暂(5~30秒)的性能波动的时候,一般的性能监控工具都很难抓住故障现场,也就很难收集对应较细粒度的诊断信息。另外,如果这种波动出现的频率很低,例如几天才一次,我们也很难人为的抓住现场,…...
女人与马做受网站/磁力多多
标砖库提供了 net/rpc 包用来实现基础的rpc调用。 net/rpc库使用encoding/gob进行编解码,支持tcp或http数据传输方式,由于其他语言不支持gob编解码方式,所以使用net/rpc库实现的RPC方法没办法进行跨语言调用。 主要有服务端和客户端。 首先…...
自己做的网站怎么添加采集模块/广东seo推广外包
我保证这是我目前写过的所有科普教程中最不科普、最底层、最不想被读、最难理解、最晦涩的一篇。但我也保证这是我目前写过的所有科普教程中最专业、最珍贵、最耗时、最尽力,最精益求精的一篇。以下正文。做3D开发(尤其是建模相关)的话&#…...
合肥响应式网站建设/爱站网站长seo综合查询
分析linux中断中常用的两个中断函数:Disable_irq(int irq)Enable_irq(int irq)第一步:●对于关中断跟踪代码到arch/arm/kernel/irq.cvoid disable_irq(unsigned int irq){struct irqdesc *desc irq_desc irq;disable_irq_nosync(irq);if (desc->act…...