使用HttpURLConnection实现多线程下载

2024-05-07 03:58

本文主要是介绍使用HttpURLConnection实现多线程下载,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

随时随地技术实战干货,获取项目源码、学习资料,请关注源代码社区公众号(ydmsq666)

HttpURLConnection继承了URLConnection,因此也可用于向指定网站发送GET请求、POST请求,而且它在URLConnection基础上提供了如下便捷方法:

实现多线程下载的步骤:

下面用一个示例来示范使用HttpURLConnection实现多线程下载。此代码来源疯狂讲义一书,该代码主要思路:在Activity中点击按钮,调用DownUtil的download()方法,在download()中启动四个线程去下载资源,每个线程负责下载自己的那部分资源,代码如下:

Activity:

package com.home.activity;import java.util.Timer;
import java.util.TimerTask;import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.EditText;
import android.widget.ProgressBar;import com.home.multithreaddown.R;
import com.home.util.DownUtil;public class MultiThreadDownActivity extends Activity {private EditText urlText;private EditText targetText;private Button downBtn;private ProgressBar bar;private DownUtil downUtil;private int mDownStatus;private Handler handler;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.main);// 获取界面中控件targetText = (EditText) findViewById(R.id.main_et_name);urlText = (EditText) findViewById(R.id.main_et_url);downBtn = (Button) findViewById(R.id.main_btn_download);bar = (ProgressBar) findViewById(R.id.main_progressBar);// 创建一个Handler对象handler = new Handler() {public void handleMessage(Message msg) {if (msg.what == 0x123) {bar.setProgress(mDownStatus);}}};downBtn.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {// 初始化DownUtil对象downUtil = new DownUtil(urlText.getText().toString(),targetText.getText().toString(), 4);try {// 开始下载downUtil.download();} catch (Exception e) {e.printStackTrace();}// 定义每秒调度获取一次系统的完成进度final Timer timer = new Timer();timer.schedule(new TimerTask() {public void run() {// 获取下载任务的完成比率double completeRate = downUtil.getCompleteRate();mDownStatus = (int) (completeRate * 100);// 发送消息通知界面更新进度条handler.sendEmptyMessage(0x123);// 下载完成后取消任务调度if (mDownStatus >= 100) {timer.cancel();}}}, 0, 100);}});}}

下载的工具类(DownUtil):

package com.home.util;import java.io.InputStream;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL;public class DownUtil {// 定义下载资源的路径private String path;// 指定所下载的文件的保存位置private String targetFile;// 定义需要使用多少线程下载资源private int threadNum;// 定义下载的文件的总大小private int fileSize;// 定义下载的线程对象private DownloadThread[] threads;public DownUtil(String path, String targetFile, int threadNum) {this.path = path;this.threadNum = threadNum;// 初始化threads数组threads = new DownloadThread[threadNum];this.targetFile = targetFile;}public void download() throws Exception {URL url = new URL(path);HttpURLConnection conn = (HttpURLConnection) url.openConnection();conn.setConnectTimeout(5 * 1000);conn.setRequestMethod("GET");conn.setRequestProperty("Accept","image/gif,image/jpeg,image/pjpeg,application/x-shockwaveflash,application/x-ms-xbap,application/xaml+xml,application/vnd.ms-xpsdocument,application/x-ms-application,application/vnd.ms-excel,application/vnd.ms-powerpoint,application/msword,*/*");conn.setRequestProperty("Accept-Language", "zh-CN");conn.setRequestProperty("Charset", "UTF-8");conn.setRequestProperty("User-Agent","Mozilla/4.0(compatible;MSIE7.0;Windows NT 5.2;Trident/4.0;.NET CLR 1.1.4322;.NET CLR 2.0.50727;.NET CLR 3.0.04506.30;.NET CLR 3.0.4506.2152;.NET CLR 3.5.30729)");conn.setRequestProperty("Connection", "Keep-Alive");// 得到文件大小fileSize = conn.getContentLength();conn.disconnect();int currentPartSize = fileSize / threadNum + 1;RandomAccessFile file = new RandomAccessFile(targetFile, "rw");// 设置本地文件的大小file.setLength(fileSize);file.close();for (int i = 0; i < threadNum; i++) {// 计算每条线程的下载的开始位置int startPos = i * currentPartSize;// 每个线程使用一个RandomAccessFile进行下载RandomAccessFile currentPart = new RandomAccessFile(targetFile,"rw");// 定位该线程的下载位置currentPart.seek(startPos);// 创建下载线程threads[i] = new DownloadThread(startPos, currentPartSize,currentPart);// 启动下载线程threads[i].start();}}/*** 获取下载完成的百分比* * @return*/public double getCompleteRate() {// 统计多条线程已经下载的总大小int sumSize = 0;for (int i = 0; i < threadNum; i++) {sumSize += threads[i].length;}// 返回已经完成的百分比return sumSize * 1.0 / fileSize;}private class DownloadThread extends Thread {// 当前线程的下载位置private int startPos;// 定义当前线程负责下载的文件大小private int currentPartSize;// 当前线程需要下载的文件块private RandomAccessFile currentPart;// 定义该线程已下载的字节数private int length = 0;public DownloadThread(int startPos, int currentPartSize,RandomAccessFile currentPart) {this.startPos = startPos;this.currentPartSize = currentPartSize;this.currentPart = currentPart;}public void run() {try {URL url = new URL(path);HttpURLConnection conn = (HttpURLConnection) url.openConnection();conn.setConnectTimeout(5 * 1000);conn.setRequestMethod("GET");conn.setRequestProperty("Accept","image/gif,image/jpeg,image/pjpeg,application/x-shockwaveflash,application/x-ms-xbap,application/xaml+xml,application/vnd.ms-xpsdocument,application/x-ms-application,application/vnd.ms-excel,application/vnd.ms-powerpoint,application/msword,*/*");conn.setRequestProperty("Accept-Language", "zh-CN");conn.setRequestProperty("Charset", "UTF-8");InputStream is = conn.getInputStream();// 跳过startPos个字符,表明该线程只下载自己负责那部分文件is.skip(startPos);byte[] by = new byte[1024];int hasRead = 0;// 读取网络数据,并写入本地文件while (length < currentPartSize&& (hasRead = is.read(by)) != -1) {currentPart.write(by, 0, hasRead);// 累计该线程下载的总大小length += hasRead;}currentPart.close();is.close();} catch (Exception e) {e.printStackTrace();}}}
}

Activity布局XML:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical" ><LinearLayoutandroid:layout_width="match_parent"android:layout_height="wrap_content"android:orientation="horizontal" ><TextViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:text="输入url:" /><EditTextandroid:id="@+id/main_et_url"android:layout_width="match_parent"android:layout_height="wrap_content" /></LinearLayout><LinearLayoutandroid:layout_width="match_parent"android:layout_height="wrap_content"android:orientation="horizontal" ><TextViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:text="输入保存的文件名:" /><EditTextandroid:id="@+id/main_et_name"android:layout_width="match_parent"android:layout_height="wrap_content" /></LinearLayout><Buttonandroid:id="@+id/main_btn_download"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="下载" /><ProgressBarandroid:id="@+id/main_progressBar"style="@android:style/Widget.ProgressBar.Horizontal"android:layout_width="match_parent"android:layout_height="wrap_content" /></LinearLayout>

权限:

<!-- 在SD卡中创建与删除文件权限 -->
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>
<!-- 向SD卡写入数据权限 -->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<!-- 授权访问网络 -->
<uses-permission android:name="android.permission.INTERNET"/>




 

这篇关于使用HttpURLConnection实现多线程下载的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

SpringBoot线程池配置使用示例详解

《SpringBoot线程池配置使用示例详解》SpringBoot集成@Async注解,支持线程池参数配置(核心数、队列容量、拒绝策略等)及生命周期管理,结合监控与任务装饰器,提升异步处理效率与系统... 目录一、核心特性二、添加依赖三、参数详解四、配置线程池五、应用实践代码说明拒绝策略(Rejected

C++ Log4cpp跨平台日志库的使用小结

《C++Log4cpp跨平台日志库的使用小结》Log4cpp是c++类库,本文详细介绍了C++日志库log4cpp的使用方法,及设置日志输出格式和优先级,具有一定的参考价值,感兴趣的可以了解一下... 目录一、介绍1. log4cpp的日志方式2.设置日志输出的格式3. 设置日志的输出优先级二、Window

Ubuntu如何分配​​未使用的空间

《Ubuntu如何分配​​未使用的空间》Ubuntu磁盘空间不足,实际未分配空间8.2G因LVM卷组名称格式差异(双破折号误写)导致无法扩展,确认正确卷组名后,使用lvextend和resize2fs... 目录1:原因2:操作3:报错5:解决问题:确认卷组名称​6:再次操作7:验证扩展是否成功8:问题已解

Qt使用QSqlDatabase连接MySQL实现增删改查功能

《Qt使用QSqlDatabase连接MySQL实现增删改查功能》这篇文章主要为大家详细介绍了Qt如何使用QSqlDatabase连接MySQL实现增删改查功能,文中的示例代码讲解详细,感兴趣的小伙伴... 目录一、创建数据表二、连接mysql数据库三、封装成一个完整的轻量级 ORM 风格类3.1 表结构

基于Python实现一个图片拆分工具

《基于Python实现一个图片拆分工具》这篇文章主要为大家详细介绍了如何基于Python实现一个图片拆分工具,可以根据需要的行数和列数进行拆分,感兴趣的小伙伴可以跟随小编一起学习一下... 简单介绍先自己选择输入的图片,默认是输出到项目文件夹中,可以自己选择其他的文件夹,选择需要拆分的行数和列数,可以通过

Python中将嵌套列表扁平化的多种实现方法

《Python中将嵌套列表扁平化的多种实现方法》在Python编程中,我们常常会遇到需要将嵌套列表(即列表中包含列表)转换为一个一维的扁平列表的需求,本文将给大家介绍了多种实现这一目标的方法,需要的朋... 目录python中将嵌套列表扁平化的方法技术背景实现步骤1. 使用嵌套列表推导式2. 使用itert

使用Docker构建Python Flask程序的详细教程

《使用Docker构建PythonFlask程序的详细教程》在当今的软件开发领域,容器化技术正变得越来越流行,而Docker无疑是其中的佼佼者,本文我们就来聊聊如何使用Docker构建一个简单的Py... 目录引言一、准备工作二、创建 Flask 应用程序三、创建 dockerfile四、构建 Docker

Python使用vllm处理多模态数据的预处理技巧

《Python使用vllm处理多模态数据的预处理技巧》本文深入探讨了在Python环境下使用vLLM处理多模态数据的预处理技巧,我们将从基础概念出发,详细讲解文本、图像、音频等多模态数据的预处理方法,... 目录1. 背景介绍1.1 目的和范围1.2 预期读者1.3 文档结构概述1.4 术语表1.4.1 核

Python使用pip工具实现包自动更新的多种方法

《Python使用pip工具实现包自动更新的多种方法》本文深入探讨了使用Python的pip工具实现包自动更新的各种方法和技术,我们将从基础概念开始,逐步介绍手动更新方法、自动化脚本编写、结合CI/C... 目录1. 背景介绍1.1 目的和范围1.2 预期读者1.3 文档结构概述1.4 术语表1.4.1 核

在Linux中改变echo输出颜色的实现方法

《在Linux中改变echo输出颜色的实现方法》在Linux系统的命令行环境下,为了使输出信息更加清晰、突出,便于用户快速识别和区分不同类型的信息,常常需要改变echo命令的输出颜色,所以本文给大家介... 目python录在linux中改变echo输出颜色的方法技术背景实现步骤使用ANSI转义码使用tpu