Java中多线程下载图片并压缩能提高效率吗


Posted in Java/Android onJuly 01, 2021
目录
  • 前言
  • 实现思路
  • 实测

前言

需求 导出Excel:本身以为是一个简单得导出,但是每行得记录文件中有一列为图片url,需要下载所有记录行对应得图片,然后压缩整个文件夹。

Java中多线程下载图片并压缩能提高效率吗

这里只做4.5.得代码讲解描述,其它也没什么好说得,话不多说上代码.

实现思路

多线程实现使用了线程池,Jdk1.8并发包下的CompletableFuture

第一步:得到基础数值

// 线程数
        Integer threadNum = 10;
        // 每条线程需要处理的图片数  
        int dataNum = imageInfoVos.size() / threadNum;
        // 写入线程数
        List<Integer> threadS = new ArrayList<>();
        for(int i=0; i<threadNum; i++){
                threadS.add(i);
        }

首先我们保存了需要下载的图片的Url列表,多线程的方式下载我们需要保证每个线程下载的图片不会重复,因此我们需要根据规则来切割保存Url列表的集合,从而保证每个线程下载属于自己的任务,上代码:

// 接上文代码
 threadS.stream().map(item -> CompletableFuture.runAsync(() ->{
                List<Image> theadItem = imageInfoVos.subList(dataNum * item,(item+1)==threadNum?imageInfoVos.size():Math.min(dataNum * (item + 1 ), imageInfoVos.size()));
                        threadDownPic(theadItem,item,dirName);
            },threadPoolTaskExecutor)).collect(Collectors.toList()).forEach(item ->{
                try {
                    item.get();
                }catch (Exception e){
                    log.error("============  多线程down执行等待异常 msg:{} =============", e.getMessage());
                }
    });

这里进行拆分讲解

使用CompletableFuture.runAsync 走异步方式,遍历item

如item=10,也就是线程数为10,则直接执行10次(有线程池的前提下)

// 使用CompletableFuture.runAsync 走异步方式,遍历item
 // 如item=10,也就是线程数为10,则直接执行10次(有线程池的前提下)
 threadS.stream().map(item -> CompletableFuture.runAsync(() ->{

规则:根据item数值通过sublist 从开始到结束,截取对应线程所需要下载的Url列表

例:dataNum为每个线程需要完成的下载数如上文 dataNum为100时

如:item=0 dataNum* item(0) =0,Math.min(dataNum * (item + 1 )=100

(item+1)==threadNum?imageInfoVos.size() 此次是为了保证最后一个线程处理最后不足的图片

根据如上规则即可得到每个线程需要下载的图片Url保证不会重复

// 根据item数值通过sublist 从开始到结束,截取对应线程所需要下载的Url列表
   // 例:dataNum为每个线程需要完成的下载数如上文 dataNum为100时
   // 如:item=0 dataNum* item(0) =0,Math.min(dataNum * (item + 1 )=100
   // 根据如上规则即可得到每个线程需要下载的图片Url保证不会重复
   // (item+1)==threadNum?imageInfoVos.size() 此次是为了保证最后一个线程处理最后不足的图片
   List<ImageInfoVo> theadItem = imageInfoVos.subList(dataNum * item,(item+1)==threadNum?imageInfoVos.size():Math.min(dataNum * (item + 1 ), imageInfoVos.size()));
   // theadItem:图片Url  item:所属下标  dirName:写入路径url
   threadDownPic(theadItem,item,dirName);

由于执行的异步方式,此处是为了线程池中所有线程都结束才能往下走,执行压缩文件步骤,这里提一嘴,如果没有手动赋予线程池,CompletableFuture默认使用ForkJoinPool.commonPool,会根据电脑核心数来指定,
比如:我本机未指定就是7个线程,执行方法时,会执行完前面7个线程任务,才会继续创建3个线程继续执行后续未完成的

},threadPoolTaskExecutor)).collect(Collectors.toList()).forEach(item ->{
                try {
                    item.get();
                }catch (Exception e){
                    log.error("============  多线程down执行等待异常 msg:{} =============", e.getMessage());
                }
            });

实测

主要代码也写完了,这种方式真的能提高效率吗?下面我贴几张测试图来说明

Java中多线程下载图片并压缩能提高效率吗

其实这种方式并没有显著的提高效率,当然这是我本机环境测试的。

效率是由网速决定,而不是由本机Cpu和io决定,比如10M带宽,一个线程一个一个顺序下载,但速度是10M,10个线程,可能每个线程的速度是1M,结果没有什么两样。

相对于网速,多线程带来的cpu以及io节省的时间几乎可以忽略,瓶颈还是在网速.

到此这篇关于Java中多线程下载图片并压缩能提高效率吗的文章就介绍到这了,更多相关Java 多线程下载提高效率内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Java/Android 相关文章推荐
总结一些Java常用的加密算法
Jun 11 Java/Android
Spring Data JPA的Audit功能审计数据库的变更
Jun 26 Java/Android
Java并发编程必备之Future机制
Jun 30 Java/Android
spring boot中nativeQuery的用法
Jul 26 Java/Android
SpringBoot实现quartz定时任务可视化管理功能
Aug 30 Java/Android
IDEA2021.2配置docker如何将springboot项目打成镜像一键发布部署
Sep 25 Java/Android
解析mybatis-plus中的resultMap简单使用
Nov 23 Java/Android
你知道Java Spring的两种事务吗
Mar 16 Java/Android
关于EntityWrapper的in用法
Mar 22 Java/Android
mapstruct的用法之qualifiedByName示例详解
Apr 06 Java/Android
前端与RabbitMQ实时消息推送未读消息小红点实现示例
Jul 23 Java/Android
Java代码规范与质量检测插件SonarLint的使用
Aug 05 Java/Android
分析ZooKeeper分布式锁的实现
Java并发编程必备之Future机制
详解Spring Boot使用系统参数表提升系统的灵活性
Jun 30 #Java/Android
浅谈resultMap的用法及关联结果集映射
Spring中bean的生命周期之getSingleton方法
每日六道java新手入门面试题,通往自由的道路
Jun 30 #Java/Android
mybatis 解决从列名到属性名的自动映射失败问题
Jun 30 #Java/Android
You might like
用Zend Encode编写开发PHP程序
2010/02/21 PHP
PHP 线程安全与非线程安全版本的区别深入解析
2013/08/06 PHP
php获取操作系统语言代码
2013/11/04 PHP
解决phpcms更换javascript的幻灯片代码调用图片问题
2014/12/26 PHP
PHP列出MySQL中所有数据库的方法
2015/03/12 PHP
php魔术方法功能与用法实例分析
2016/10/19 PHP
Javascript 生成指定范围数值随机数
2009/01/09 Javascript
WEB页子窗口(showModalDialog和showModelessDialog)使用说明
2009/10/25 Javascript
JavaScript 学习初步 入门教程
2010/03/25 Javascript
js nextSibling属性和previousSibling属性概述及使用注意
2013/02/16 Javascript
获取内联和链接中的样式(js代码)
2013/04/11 Javascript
JavaScript图片放大技术(放大镜)实现代码分享
2013/11/14 Javascript
JS动态修改表格cellPadding和cellSpacing的方法
2015/03/31 Javascript
用Move.js配合创建CSS3动画的入门指引
2015/07/22 Javascript
JS实现的鼠标跟随代码(卡通手型点击效果)
2015/10/26 Javascript
JS生成和下载二维码的代码
2016/12/07 Javascript
详解element-ui中表单验证的三种方式
2019/09/18 Javascript
JS如何在数组指定位置插入元素
2020/03/10 Javascript
下载糗事百科的内容_python版
2008/12/07 Python
仅用500行Python代码实现一个英文解析器的教程
2015/04/02 Python
在Python程序中进行文件读取和写入操作的教程
2015/04/28 Python
python通过加号运算符操作列表的方法
2015/07/28 Python
Python下调用Linux的Shell命令的方法
2018/06/12 Python
python3 下载网络图片代码实例
2019/08/27 Python
在Django下测试与调试REST API的方法详解
2019/08/29 Python
Python datetime 格式化 明天,昨天实例
2020/03/02 Python
2020最新pycharm汉化安装(python工程狮亲测有效)
2020/04/26 Python
Python实现清理微信僵尸粉功能示例【基于itchat模块】
2020/05/29 Python
Canvas在超级玛丽游戏中的应用详解
2021/02/06 HTML / CSS
美国最大的船只买卖在线市场:Boat Trader
2018/08/04 全球购物
英国最大的汽车配件在线商店:Euro Car Parts
2019/09/30 全球购物
大学生职业生涯规划书范文
2014/01/04 职场文书
群众路线教育实践活动民主生活会个人检查对照思想汇报
2014/10/04 职场文书
三方股份合作协议书
2014/10/13 职场文书
中标通知书格式
2015/04/17 职场文书
2016廉洁从政心得体会
2016/01/19 职场文书