Java中多線程下載圖片并壓縮能提高效率嗎
前言
需求 導(dǎo)出Excel:本身以為是一個(gè)簡單得導(dǎo)出,但是每行得記錄文件中有一列為圖片url,需要下載所有記錄行對應(yīng)得圖片,然后壓縮整個(gè)文件夾。

這里只做4.5.得代碼講解描述,其它也沒什么好說得,話不多說上代碼.
實(shí)現(xiàn)思路
多線程實(shí)現(xiàn)使用了線程池,Jdk1.8并發(fā)包下的CompletableFuture
第一步:得到基礎(chǔ)數(shù)值
// 線程數(shù)
Integer threadNum = 10;
// 每條線程需要處理的圖片數(shù)
int dataNum = imageInfoVos.size() / threadNum;
// 寫入線程數(shù)
List<Integer> threadS = new ArrayList<>();
for(int i=0; i<threadNum; i++){
threadS.add(i);
}
首先我們保存了需要下載的圖片的Url列表,多線程的方式下載我們需要保證每個(gè)線程下載的圖片不會重復(fù),因此我們需要根據(jù)規(guī)則來切割保存Url列表的集合,從而保證每個(gè)線程下載屬于自己的任務(wù),上代碼:
// 接上文代碼
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執(zhí)行等待異常 msg:{} =============", e.getMessage());
}
});
這里進(jìn)行拆分講解
使用CompletableFuture.runAsync 走異步方式,遍歷item
如item=10,也就是線程數(shù)為10,則直接執(zhí)行10次(有線程池的前提下)
// 使用CompletableFuture.runAsync 走異步方式,遍歷item
// 如item=10,也就是線程數(shù)為10,則直接執(zhí)行10次(有線程池的前提下)
threadS.stream().map(item -> CompletableFuture.runAsync(() ->{
規(guī)則:根據(jù)item數(shù)值通過sublist 從開始到結(jié)束,截取對應(yīng)線程所需要下載的Url列表
例:dataNum為每個(gè)線程需要完成的下載數(shù)如上文 dataNum為100時(shí)
如:item=0 dataNum* item(0) =0,Math.min(dataNum * (item + 1 )=100
(item+1)==threadNum?imageInfoVos.size() 此次是為了保證最后一個(gè)線程處理最后不足的圖片
根據(jù)如上規(guī)則即可得到每個(gè)線程需要下載的圖片Url保證不會重復(fù)
// 根據(jù)item數(shù)值通過sublist 從開始到結(jié)束,截取對應(yīng)線程所需要下載的Url列表 // 例:dataNum為每個(gè)線程需要完成的下載數(shù)如上文 dataNum為100時(shí) // 如:item=0 dataNum* item(0) =0,Math.min(dataNum * (item + 1 )=100 // 根據(jù)如上規(guī)則即可得到每個(gè)線程需要下載的圖片Url保證不會重復(fù) // (item+1)==threadNum?imageInfoVos.size() 此次是為了保證最后一個(gè)線程處理最后不足的圖片 List<ImageInfoVo> theadItem = imageInfoVos.subList(dataNum * item,(item+1)==threadNum?imageInfoVos.size():Math.min(dataNum * (item + 1 ), imageInfoVos.size())); // theadItem:圖片Url item:所屬下標(biāo) dirName:寫入路徑url threadDownPic(theadItem,item,dirName);
由于執(zhí)行的異步方式,此處是為了線程池中所有線程都結(jié)束才能往下走,執(zhí)行壓縮文件步驟,這里提一嘴,如果沒有手動賦予線程池,CompletableFuture默認(rèn)使用ForkJoinPool.commonPool,會根據(jù)電腦核心數(shù)來指定,
比如:我本機(jī)未指定就是7個(gè)線程,執(zhí)行方法時(shí),會執(zhí)行完前面7個(gè)線程任務(wù),才會繼續(xù)創(chuàng)建3個(gè)線程繼續(xù)執(zhí)行后續(xù)未完成的
},threadPoolTaskExecutor)).collect(Collectors.toList()).forEach(item ->{
try {
item.get();
}catch (Exception e){
log.error("============ 多線程down執(zhí)行等待異常 msg:{} =============", e.getMessage());
}
});
實(shí)測
主要代碼也寫完了,這種方式真的能提高效率嗎?下面我貼幾張測試圖來說明

其實(shí)這種方式并沒有顯著的提高效率,當(dāng)然這是我本機(jī)環(huán)境測試的。
效率是由網(wǎng)速決定,而不是由本機(jī)Cpu和io決定,比如10M帶寬,一個(gè)線程一個(gè)一個(gè)順序下載,但速度是10M,10個(gè)線程,可能每個(gè)線程的速度是1M,結(jié)果沒有什么兩樣。
相對于網(wǎng)速,多線程帶來的cpu以及io節(jié)省的時(shí)間幾乎可以忽略,瓶頸還是在網(wǎng)速.
到此這篇關(guān)于Java中多線程下載圖片并壓縮能提高效率嗎的文章就介紹到這了,更多相關(guān)Java 多線程下載提高效率內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java實(shí)現(xiàn)轉(zhuǎn)換圖片格式的示例代碼
在日常的軟件開發(fā)中,處理圖像文件是一項(xiàng)常見任務(wù),這篇文章將實(shí)現(xiàn)實(shí)現(xiàn)一個(gè)簡單的Java程序,用于將一種圖片格式轉(zhuǎn)換為另一種格式,需要的可以了解下2025-02-02
vue+springboot讀取git的markdown文件并展示功能
Markdown-it 是一個(gè)用于解析和渲染 Markdown 標(biāo)記語言的 JavaScript 庫,使用 Markdown-it,你可以將 Markdown 文本解析為 HTML 輸出,并且可以根據(jù)需要添加功能、擴(kuò)展語法或修改解析行為,本文介紹vue+springboot讀取git的markdown文件并展示,感興趣的朋友一起看看吧2024-01-01
IDEA 的基本介紹使用及斷點(diǎn)調(diào)試技巧
IDEA 是 JetBrains 公司的產(chǎn)品,總部位于捷克的首都布拉格,IDEA在業(yè)界被公認(rèn)為最好的 Java 開發(fā)工具,今天通過本文給大家介紹IDEA 的基本介紹使用及斷點(diǎn)調(diào)試技巧,感興趣的朋友跟隨小編一起看看吧2021-11-11
java大話之創(chuàng)建型設(shè)計(jì)模式教程示例
這篇文章主要為大家介紹了java大話之創(chuàng)建型設(shè)計(jì)模式教程示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-02-02
注入jar包里的對象,用@autowired的實(shí)例
這篇文章主要介紹了注入jar包里的對象,用@autowired的實(shí)例,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-09-09
Maven Plugins報(bào)錯(cuò)的解決方法
本文主要介紹了Maven Plugins報(bào)錯(cuò)的解決方法,文中通過圖文介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-02-02
Java中MessageDigest來實(shí)現(xiàn)數(shù)據(jù)加密的方法
這篇文章主要介紹了Java中MessageDigest來實(shí)現(xiàn)數(shù)據(jù)加密的方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05

