在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

Spring多線程異步上傳圖片、處理水印、縮略圖

jf_ro2CN3Fa ? 來源:芋道源碼 ? 作者:芋道源碼 ? 2022-12-12 16:02 ? 次閱讀


使用環(huán)境

  • SpringBoot+FastDfs+thumbnailator
  • fdfs環(huán)境自己搞吧

基于 Spring Boot + MyBatis Plus + Vue & Element 實(shí)現(xiàn)的后臺管理系統(tǒng) + 用戶小程序,支持 RBAC 動(dòng)態(tài)權(quán)限、多租戶、數(shù)據(jù)權(quán)限、工作流、三方登錄、支付、短信、商城等功能

  • 項(xiàng)目地址:https://github.com/YunaiV/ruoyi-vue-pro
  • 視頻教程:https://doc.iocoder.cn/video/

thumbnailator

maven依賴:

<dependency>
<groupId>net.coobirdgroupId>
<artifactId>thumbnailatorartifactId>
<version>0.4.8version>
dependency>

工具類:

importnet.coobird.thumbnailator.Thumbnails;
importnet.coobird.thumbnailator.geometry.Positions;
importorg.springframework.stereotype.Component;
importjavax.imageio.ImageIO;
importjava.io.File;
importjava.io.IOException;

@Component
publicclassPictureUtil{

/**
*水印圖片
*/
privatestaticFilemarkIco=null;

//開機(jī)靜態(tài)加載水印圖片
static{
try{
markIco=newFile(newFile("").getCanonicalPath()+"/icon.png");
LogUtil.info(PictureUtil.class,"水印圖片加載"+(markIco.exists()?"成功":"失敗"));
}catch(Exceptione){
}
}

/**
*加水印
*/
publicvoidphotoMark(FilesourceFile,FiletoFile)throwsIOException{
Thumbnails.of(sourceFile)
.size(600,450)//尺寸
.watermark(Positions.BOTTOM_CENTER/*水印位置:中央靠下*/,
ImageIO.read(markIco),0.7f/*質(zhì)量,越大質(zhì)量越高(1)*/)
//.outputQuality(0.8f)
.toFile(toFile);//保存為哪個(gè)文件
}

/**
*生成圖片縮略圖
*/
publicvoidphotoSmaller(FilesourceFile,FiletoFile)throwsIOException{
Thumbnails.of(sourceFile)
.size(200,150)//尺寸
//.watermark(Positions.CENTER,ImageIO.read(markIco),0.1f)
.outputQuality(0.4f)//縮略圖質(zhì)量
.toFile(toFile);
}

/**
*生成視頻縮略圖(這塊還沒用到呢)
*/
publicvoidphotoSmallerForVedio(FilesourceFile,FiletoFile)throwsIOException{
Thumbnails.of(sourceFile)
.size(440,340)
.watermark(Positions.BOTTOM_CENTER,ImageIO.read(markIco),0.1f)
.outputQuality(0.8f)
.toFile(toFile);
}
}

這個(gè)插件很好用,只需集成調(diào)用即可,我記得我還試過另外幾個(gè),需要另外在linux下配置.so文件的依賴等等,查了半天也沒弄明白,很麻煩,這個(gè)方便。

這個(gè)插件又很不好用,必須要先調(diào)整尺寸,才能加水印,而且調(diào)整尺寸簡直是負(fù)壓縮。壓了分辨率圖片還能變大那種。但是簡單嘛,這塊不是重點(diǎn)。

基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 實(shí)現(xiàn)的后臺管理系統(tǒng) + 用戶小程序,支持 RBAC 動(dòng)態(tài)權(quán)限、多租戶、數(shù)據(jù)權(quán)限、工作流、三方登錄、支付、短信、商城等功能

  • 項(xiàng)目地址:https://github.com/YunaiV/yudao-cloud
  • 視頻教程:https://doc.iocoder.cn/video/

線程池

使用springboot線程池,方便易用,只需配置和加注解即可。

importorg.springframework.context.annotation.Bean;
importorg.springframework.context.annotation.Configuration;
importorg.springframework.core.task.TaskExecutor;
importorg.springframework.scheduling.annotation.EnableAsync;
importorg.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;
importjava.util.concurrent.ThreadPoolExecutor;

@Configuration
@EnableAsync
publicclassPoolConfig{
@Bean//returnnewAsyncResult<>(res);
publicTaskExecutortaskExecutor(){
ThreadPoolTaskExecutorexecutor=newThreadPoolTaskExecutor();
executor.initialize();//設(shè)置核心線程數(shù)
executor.setCorePoolSize(4);//設(shè)置最大線程數(shù)
executor.setMaxPoolSize(32);//設(shè)置隊(duì)列容量
executor.setQueueCapacity(512);//設(shè)置線程活躍時(shí)間(秒)
executor.setKeepAliveSeconds(60);//設(shè)置默認(rèn)線程名稱
executor.setThreadNamePrefix("ThreadPool-");//設(shè)置拒絕策略
executor.setRejectedExecutionHandler(newThreadPoolExecutor.CallerRunsPolicy());//等待所有任務(wù)結(jié)束后再關(guān)閉線程池
executor.setWaitForTasksToCompleteOnShutdown(true);
returnexecutor;
}
}

避坑知識點(diǎn):配置springboot線程池,類上需要@Configuration@EnableAsync這兩個(gè)注解,實(shí)際調(diào)用時(shí),需要遵守一個(gè)規(guī)則,即在調(diào)用的方法的類上必須使用注解@EnableAsync,調(diào)用一個(gè)帶有@Async的方法。

比如A類使用了注解@EnableAsync 在A類中調(diào)用B類的有@Async的方法,只有這樣多線程才生效,A類內(nèi)調(diào)用A類的@Async方法不生效。可以理解為Controller層使用@EnableAsync注解,Service層方法上標(biāo)注@Async。這樣在Controller層調(diào)用的Service方法會(huì)從線程池調(diào)用線程來執(zhí)行。

異步邏輯:為什么要用多線程?

abafc9dc-794f-11ed-8abf-dac502259ad0.png

我畫了一張簡單的示意圖,在這個(gè)項(xiàng)目中,客戶端一次上傳10多張圖片,每個(gè)圖片單獨(dú)上傳,等待所有圖片上傳返回200后,繼續(xù)執(zhí)行操作,如果一步一步處理,客戶端需等待服務(wù)器處理完所有邏輯,這樣浪費(fèi)沒必要的時(shí)間。顧使用異步操作,客戶端只需上傳圖片,無需等待服務(wù)器處理(我們服務(wù)器很辣雞,一個(gè)10M的圖可能要搞10多秒,見笑)

業(yè)務(wù)代碼

@ApiOperation("上傳業(yè)務(wù)圖片")
@PostMapping("/push/photo/{id}/{name}")
publicRpushHousingPhotoMethod(
@ApiParam("SourceId")@PathVariableIntegerid,
@ApiParam("圖片名稱不約束,可不填則使用原名,可使用隨機(jī)碼或原名稱,但必須帶擴(kuò)展名")@PathVariable(required=false)Stringname,
@RequestParamMultipartFilefile)throwsInterruptedException,ExecutionException,IOException{
StringfileName=file.getOriginalFilename();
Stringext=StringUtils.substring(fileName,fileName.lastIndexOf('.'),fileName.length());
FiletempPhoto=File.createTempFile(UUIDUtil.make32BitUUID(),ext);
file.transferTo(tempPhoto);//轉(zhuǎn)儲臨時(shí)文件
service.pushPhoto(id,name,tempPhoto);
returnnewR();
}

業(yè)務(wù)代碼里隱藏了一些項(xiàng)目相關(guān)的信息,就是某些名改了,嗯。

可以看到,使用StringUtils.substring(fileName, fileName.lastIndexOf(’.’),fileName.length());這句代碼,調(diào)用apache.common.lang3工具類獲取出了擴(kuò)展名,因?yàn)閿U(kuò)展名對圖片處理工具類有用,他通過擴(kuò)展名識別圖片格式,所以這個(gè)必須有,如代碼,生成了一個(gè)使用隨機(jī)碼命名,但帶有.png擴(kuò)展名的臨時(shí)文件,保存在默認(rèn)臨時(shí)路徑以供處理。File.createTempFile(UUIDUtil.make32BitUUID(), ext);是生成臨時(shí)文件的方法,UUIDUtil也很簡單,我貼出來吧,省著還要找

注意:controller類上需要標(biāo)注注解@EnableAsync

/**
*生成一個(gè)32位無橫杠的UUID
*/
publicsynchronizedstaticStringmake32BitUUID(){
returnUUID.randomUUID().toString().replace("-","");
}

避坑知識點(diǎn):Spring使用MultipartFile接收文件,但不能直接把MultipartFile傳下去處理,而是保存為臨時(shí)文件,并不是多此一舉。因?yàn)?code style="font-size:14px;padding:2px 4px;margin-right:2px;margin-left:2px;color:rgb(30,107,184);background-color:rgba(27,31,35,.05);font-family:'Operator Mono', Consolas, Monaco, Menlo, monospace;">MultipartFile也是臨時(shí)文件,他的銷毀時(shí)間是你這個(gè)Controller層方法return的時(shí)候。

如果不使用異步,是可以在調(diào)用的方法里去處理MultipartFile文件的,但如果使用異步處理,肯定是這邊線程還沒處理完,那邊Controller層已經(jīng)return了,這個(gè)MultipartFile就被刪除了,于是你的異步線程就找不到這張圖了。那還處理個(gè)啥,對吧。所以需要手動(dòng)保存為自己創(chuàng)建的臨時(shí)文件,再在線程中處理完把他刪掉。

貼Service層Impl實(shí)現(xiàn)類代碼

@Async
publicvoidpushHousingPhoto(Integerid,Stringname,Filefile)throwsInterruptedException,ExecutionException,IOException{
//存儲FDFS表id
LongstartTime=System.currentTimeMillis();
Integer[]numb=fastDfsService.upLoadPhoto(StringUtils.isBlank(name)?file.getName():name,file).get();
SourcePhotosContextcontext=newSourcePhotosContext();
context.setSourceId(id);
context.setNumber(numb[0]);
context.setNumber2(numb[1]);
//保存圖片關(guān)系
sourcePhotosContextService.insertNew(context);
LongendTime=System.currentTimeMillis();
LogUtil.info(this.getClass(),"source["+id+"]綁定圖片["+name+"]成功,內(nèi)部處理耗時(shí)["+(endTime-startTime)+"ms]");
//returnnewR();
}

這里的number和number2分別是帶水印的原圖和縮略圖,context是個(gè)表,用來存圖片和縮略圖對應(yīng)fdfs路徑的,就不貼了。可見這個(gè)方法上帶有注解@Async 所以整個(gè)方法會(huì)異步執(zhí)行。

加水印處理寫到fdfs的service里了,這樣不算規(guī)范,可以不要學(xué)我:

@Override
publicFutureupLoadPhoto(StringfileName,MultipartFilefile)throwsIOException{
Stringext=StringUtils.substring(fileName,fileName.lastIndexOf('.'));
//創(chuàng)建臨時(shí)文件
FilesourcePhoto=File.createTempFile(UUIDUtil.make32BitUUID(),ext);
file.transferTo(sourcePhoto);
returnupLoadPhoto(fileName,sourcePhoto);
}

@Override
publicFutureupLoadPhoto(StringfileName,FilesourcePhoto)throwsIOException{
Stringext=StringUtils.substring(fileName,fileName.lastIndexOf('.'));
//創(chuàng)建臨時(shí)文件
FilemarkedPhoto=File.createTempFile(UUIDUtil.make32BitUUID(),ext);
FilesmallerPhoto=File.createTempFile(UUIDUtil.make32BitUUID(),ext);
//加水印縮圖
pictureUtil.photoMark(sourcePhoto,markedPhoto);
pictureUtil.photoSmaller(markedPhoto,smallerPhoto);
//上傳
IntegermarkedPhotoNumber=upLoadPhotoCtrl(fileName,markedPhoto);
IntegersmallerPhotoNumber=upLoadPhotoCtrl("mini_"+fileName,smallerPhoto);
//刪除臨時(shí)文件
sourcePhoto.delete();
markedPhoto.delete();
smallerPhoto.delete();
Integer[]res=newInteger[]{markedPhotoNumber,smallerPhotoNumber};
returnnewAsyncResult(res);
}

使用了方法重載,一個(gè)調(diào)用了另一個(gè),方便以后處理MultipartFile和File格式的圖片都能使用,可以見到使用了Future這個(gè)東西作為返回值,完全可以不這么做,正常返回就行。我懶得改了,這也是不斷探索多線程處理圖片的過程中,遺留下來的東西。

在service中fastDfsService.upLoadPhoto(StringUtils.isBlank(name) ? file.getName() : name, file).get()這句就是得到了這個(gè)future的內(nèi)容,可以去掉.get()Future<>。可見這一個(gè)小小的異步功能,其實(shí)走過了很多彎路。future其實(shí)是異步調(diào)用方法時(shí),從.get()等待異步處理的結(jié)果,等待得到結(jié)果后獲取內(nèi)容并執(zhí)行。現(xiàn)在使用spring線程池處理,已經(jīng)不需要這樣做了。

以上,希望你在實(shí)現(xiàn)這個(gè)功能時(shí)可以少走彎路。

附總體示意圖:

abd5ea90-794f-11ed-8abf-dac502259ad0.png

審核編輯 :李倩

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報(bào)投訴
  • 多線程
    +關(guān)注

    關(guān)注

    0

    文章

    279

    瀏覽量

    20312
  • spring
    +關(guān)注

    關(guān)注

    0

    文章

    340

    瀏覽量

    14891

原文標(biāo)題:Spring 多線程異步上傳圖片、處理水印、縮略圖

文章出處:【微信號:芋道源碼,微信公眾號:芋道源碼】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦
    熱點(diǎn)推薦

    鴻蒙NEXT上傳圖片功能PhotoViewPicker核心功能解析

    */ isOriginalPhoto: boolean; } 獲取到資源后,通常需要進(jìn)行以下處理: 使用文件 URI 讀取文件內(nèi)容 進(jìn)行必要的格式轉(zhuǎn)換(如壓縮圖片上傳到服務(wù)器或保存到本地 // 這里以oss
    發(fā)表于 06-06 15:00

    工控一體機(jī)多線程任務(wù)調(diào)度優(yōu)化:聚徽分享破解工業(yè)復(fù)雜流程高效協(xié)同密碼

    在當(dāng)今工業(yè) 4.0 的浪潮下,工業(yè)生產(chǎn)正朝著高度自動(dòng)化、智能化的方向大步邁進(jìn)。生產(chǎn)流程日益復(fù)雜,眾多任務(wù)需要同時(shí)、高效地協(xié)同執(zhí)行,這對工業(yè)控制系統(tǒng)的核心 —— 工控一體機(jī)提出了前所未有的挑戰(zhàn)。多線程
    的頭像 發(fā)表于 05-28 14:06 ?99次閱讀

    鴻蒙開發(fā)實(shí)現(xiàn)圖片上傳上傳用戶頭像)

    }) }) 2. 拷貝圖片到緩存目錄 當(dāng)前上傳應(yīng)用文件功能,僅支持上傳應(yīng)用緩存文件路徑(cacheDir)下的文件。 使用上傳下載模塊,需聲明權(quán)限:ohos.permission
    發(fā)表于 05-24 23:09

    一種實(shí)時(shí)多線程VSLAM框架vS-Graphs介紹

    針對現(xiàn)有VSLAM系統(tǒng)語義表達(dá)不足、地圖可解釋性差的問題,本文提出vS-Graphs,一種實(shí)時(shí)多線程VSLAM框架。該方案顯著提升了重建地圖的語義豐富度、可解釋性及定位精度。實(shí)驗(yàn)表明
    的頭像 發(fā)表于 04-19 14:07 ?283次閱讀
    一種實(shí)時(shí)<b class='flag-5'>多線程</b>VSLAM框架vS-Graphs介紹

    請問如何在Python中實(shí)現(xiàn)多線程與多進(jìn)程的協(xié)作?

    大家好!我最近在開發(fā)一個(gè)Python項(xiàng)目時(shí),需要同時(shí)處理多個(gè)任務(wù),且每個(gè)任務(wù)需要不同的計(jì)算資源。我想通過多線程和多進(jìn)程的組合來實(shí)現(xiàn)并發(fā),但遇到了一些問題。 具體來說,我有兩個(gè)任務(wù),一個(gè)是I/O密集型
    發(fā)表于 03-11 06:57

    請問rt-thread studio如何進(jìn)行多線程編譯?

    ,使用的是5800h+32g內(nèi)存+sn550 ssd,開啟16線程編譯時(shí)cpu的占用率也只能到30%,編譯完整個(gè)工程需要3分鐘 感覺多線程編譯設(shè)置沒有生效,有辦法提高編譯速度嗎
    發(fā)表于 02-19 08:30

    TFP401作為HDMI的解碼芯片,DE信號始終解析不正常是怎么回事?

    ,而且不管選用什么分辨率的輸入信號,通過FPGA抓取信號發(fā)現(xiàn),這段高電平的長度均為36個(gè)像素時(shí)鐘周期,且真正的DE信號,也比理應(yīng)真實(shí)的多兩個(gè)像素時(shí)鐘周期。 FPGA抓取信號如圖(縮略圖,與示波器抓取信號一致) 兩個(gè)放大圖,理應(yīng)抓取到的正確列分辨率為1280,實(shí)際為1282 感謝各位了!
    發(fā)表于 12-20 07:02

    socket 多線程編程實(shí)現(xiàn)方法

    在現(xiàn)代網(wǎng)絡(luò)編程中,多線程技術(shù)被廣泛應(yīng)用于提高服務(wù)器的并發(fā)處理能力。Socket編程是網(wǎng)絡(luò)通信的基礎(chǔ),而將多線程技術(shù)應(yīng)用于Socket編程,可以顯著提升服務(wù)器的性能。 多線程編程的基本概
    的頭像 發(fā)表于 11-12 14:16 ?899次閱讀

    Spring事務(wù)實(shí)現(xiàn)原理

    作者:京東零售 范錫軍 1、引言 springspring-tx模塊提供了對事務(wù)管理支持,使用spring事務(wù)可以讓我們從復(fù)雜的事務(wù)處理中得到解脫,無需要去
    的頭像 發(fā)表于 11-08 10:10 ?1118次閱讀
    <b class='flag-5'>Spring</b>事務(wù)實(shí)現(xiàn)原理

    Python中多線程和多進(jìn)程的區(qū)別

    Python作為一種高級編程語言,提供了多種并發(fā)編程的方式,其中多線程與多進(jìn)程是最常見的兩種方式之一。在本文中,我們將探討Python中多線程與多進(jìn)程的概念、區(qū)別以及如何使用線程池與進(jìn)程池來提高并發(fā)執(zhí)行效率。
    的頭像 發(fā)表于 10-23 11:48 ?924次閱讀
    Python中<b class='flag-5'>多線程</b>和多進(jìn)程的區(qū)別

    Spring Cloud Gateway網(wǎng)關(guān)框架

    Spring Cloud Gateway網(wǎng)關(guān)框架 本軟件微服務(wù)架構(gòu)中采用Spring Cloud Gateway網(wǎng)關(guān)控制框架,Spring Cloud Gateway是Spring C
    的頭像 發(fā)表于 08-22 09:58 ?688次閱讀
    <b class='flag-5'>Spring</b> Cloud Gateway網(wǎng)關(guān)框架

    一文掌握Python多線程

    使用線程可以把占據(jù)長時(shí)間的程序中的任務(wù)放到后臺去處理
    的頭像 發(fā)表于 08-05 15:46 ?1195次閱讀

    ESP32會(huì)不會(huì)有多線程問題,需要加鎖嗎?

    ESP32會(huì)不會(huì)有多線程問題,需要加鎖嗎
    發(fā)表于 07-19 08:05

    FPGA異步信號處理方法

    FPGA(現(xiàn)場可編程門陣列)在處理異步信號時(shí),需要特別關(guān)注信號的同步化、穩(wěn)定性以及潛在的亞穩(wěn)態(tài)問題。由于異步信號可能來自不同的時(shí)鐘域或外部設(shè)備,其到達(dá)時(shí)間和頻率可能不受FPGA內(nèi)部時(shí)鐘控制,因此
    的頭像 發(fā)表于 07-17 11:10 ?1787次閱讀

    多線程設(shè)計(jì)模式到對 CompletableFuture 的應(yīng)用

    最近在開發(fā) 延保服務(wù) 頻道頁時(shí),為了提高查詢效率,使用到了多線程技術(shù)。為了對多線程方案設(shè)計(jì)有更加充分的了解,在業(yè)余時(shí)間讀完了《圖解 Java 多線程設(shè)計(jì)模式》這本書,覺得收獲良多。本篇文章將介紹其中
    的頭像 發(fā)表于 06-26 14:18 ?624次閱讀
    從<b class='flag-5'>多線程</b>設(shè)計(jì)模式到對 CompletableFuture 的應(yīng)用
    主站蜘蛛池模板: 成人黄色激情网 | 伊人网络| 国产亚洲精品在天天在线麻豆 | 欧美精品video | 国产色中色 | 久久精品人人爽人人爽快 | 曰本又色又爽又黄三级视频 | 正在播放淫亚洲 | 亚洲日本在线观看视频 | 国产精品久久新婚兰兰 | 天天爽夜夜爽免费看 | 欧美色欧美色 | 高清一级做a爱视频免费 | 亚洲高清免费视频 | 玖玖草在线观看 | 久久精品免费观看视频 | 亚洲高清国产一线久久 | 黄频网站免费大全在线观看 | 久久青草91线频免费观看 | 国模娜娜扒开嫩木耳 | 国产永久视频夜色资源网 | 国产片在线 | 婷婷综合色 | 特黄特色三级在线播放 | 天天视频观看 | 午夜精品久久久久久久第一页 | 美女国产 | 免费国产午夜在线观看 | www.亚洲天堂| 色yeye在线观看 | 69pao强力打造在线 | 特级深夜a级毛片免费观看 特级生活片 | 成人在线视频网 | 91插插视频 | 日本黄色片在线播放 | 韩国三级理论在线看中文字幕 | 色综合天天综合网看在线影院 | 最好看的2019中文字幕免费高清 | 天天爱天天做天天干 | 精品99久久 | 在线精品视频成人网 |