Springboot集成GraphicsMagick
以什么方式集成?
JNI / 命令行(im4java)
在im4java官網(wǎng)中提到:
翻譯過(guò)來(lái)就是: 從Java內(nèi)部使用JNI運(yùn)行本機(jī)代碼始終會(huì)帶來(lái)其他風(fēng)險(xiǎn),對(duì)于長(zhǎng)時(shí)間運(yùn)行的進(jìn)程(通常是Web應(yīng)用程序服務(wù)器)尤其危險(xiǎn)。內(nèi)存損壞或分段錯(cuò)誤(可能由故意操縱的圖像觸發(fā))可能會(huì)使整個(gè)服務(wù)器癱瘓。
所以我們選擇使用命令行的方式進(jìn)行調(diào)用。
項(xiàng)目集成
1、將gm命令行工具引入到項(xiàng)目中
在SpringBoot集成Linux可執(zhí)行命令的時(shí)候,我們將可執(zhí)行文件放在了項(xiàng)目的resource目錄下:
這里需要有一步操作就是將文件復(fù)制到宿主機(jī):
private void initGM() throws Exception { String osName = System.getProperty("os.name").toLowerCase(); log.info("os name: {}", osName); String gmPath; if (osName.contains("mac")) { gmPath = "gm/mac/gm"; } else if (osName.contains("linux")) { // 初始化容器的環(huán)境 initPodEnv(); gmPath = "gm/linux/gm"; } else { throw new RuntimeException("非法操作系統(tǒng):"+osName); } InputStream fisInJar = new ClassPathResource(gmPath).getInputStream(); File file = File.createTempFile("GraphicsMagick", "_gm"); file.setExecutable(true); GM_PATH = file.getAbsolutePath(); //將jar包里的gm復(fù)制到操作系統(tǒng)的目錄里 OutputStream fosInOs = new FileOutputStream(file); byte[] buffer = new byte[1024]; int readLength = fisInJar.read(buffer); while (readLength != -1) { fosInOs.write(buffer, 0, readLength); readLength = fisInJar.read(buffer); } IOUtils.closeQuietly(fosInOs); IOUtils.closeQuietly(fisInJar); log.info("gm初始化完畢"); }
2、在項(xiàng)目啟動(dòng)的時(shí)候自動(dòng)初始化環(huán)境
下面只對(duì)Linux進(jìn)行了自動(dòng)化環(huán)境安裝,mac環(huán)境主要是本地開(kāi)發(fā),自己安裝環(huán)境即可:
/** * 初始化容器的環(huán)境 * * 安裝gm所依賴的庫(kù) */ private void initPodEnv() throws Exception { log.info("============ start init pod env ============"); Process exec1 = Runtime.getRuntime().exec("yum install -y gcc make"); this.printLog(exec1); log.info("cmd 1 exec success"); Process exec2 = Runtime.getRuntime().exec("yum install -y libpng-devel libjpeg-devel libtiff-devel jasper-devel freetype-devel libtool-ltdl-devel*"); this.printLog(exec2); log.info("cmd 2 exec success"); // 打水印時(shí)缺少依賴 Process exec3 = Runtime.getRuntime().exec("yum -y install ghostscript"); this.printLog(exec3); log.info("cmd 3 exec success"); log.info("============ init pod env success ============"); }
3、gm進(jìn)程池化
想象下,如果在每次進(jìn)行圖片處理都去 fork gm子進(jìn)程,不僅代價(jià)大,而且在高并發(fā)情況下,容易造成子進(jìn)程過(guò)多,導(dǎo)致系統(tǒng)負(fù)載飆高,上下文切換頻繁。
所以將 gm進(jìn)程 池化是很有必要的。
前提: gm提供batch批量模式,運(yùn)行在此模式下的gm進(jìn)程,會(huì)一直讀取標(biāo)準(zhǔn)輸入,逐行接收命令實(shí)時(shí)進(jìn)行處理。
池化思路: 預(yù)先 fork 一批 gm 子進(jìn)程,每次要運(yùn)行命令時(shí),從子進(jìn)程池中挑選一個(gè)子進(jìn)程,進(jìn)行圖片處理,處理完畢后歸還連接。
具體架構(gòu):
/** * GM 進(jìn)程池參數(shù) */ @ConfigurationProperties(prefix = "gm.pool") @Data public class GMPoolProperties { /** * 連接池最大活躍數(shù) */ private int maxActive = 4; /** * 連接池最大空閑連接數(shù) */ private int maxIdle = 4; /** * 連接池最小空閑連接數(shù) */ private int minIdle = 2; /** * 資源池中資源最小空閑時(shí)間(單位為毫秒),達(dá)到此值后空閑資源將被移 */ private long minEvictableIdleTimeMillis = 300000L; /** * 連接池連接用盡后執(zhí)行的動(dòng)作 */ private WhenExhaustedAction whenExhaustedAction = WhenExhaustedAction.BLOCK; /** * 連接池沒(méi)有對(duì)象返回時(shí),最大等待時(shí)間(毫秒) */ private long maxWait = 5000; /** * 定時(shí)對(duì)線程池中空閑的鏈接進(jìn)行校驗(yàn) */ private boolean testWhileIdle = false; /** * 空閑資源的檢測(cè)周期(單位為毫秒) */ private long timeBetweenEvictionRunsMillis = 10000L; }
性能初測(cè)
1、單線程測(cè)試: 單線程循環(huán)100次
技術(shù) | 耗時(shí) | 平均耗時(shí) |
---|---|---|
GraphicsMagick + im4java | 2110 ms | 21 ms |
GraphicsMagick + im4java + 池化技術(shù) | 1478 ms | 15 ms |
總結(jié):性能提升約29%
2、多線程并發(fā)測(cè)試: 并發(fā)100個(gè)線程請(qǐng)求
技術(shù) | 耗時(shí) | 平均耗時(shí) |
---|---|---|
GraphicsMagick + im4java | 37901 ms | 379 ms |
GraphicsMagick + im4java + 池化技術(shù) | 22456 ms | 224 ms |
總結(jié):性能提升約41%
寫(xiě)在最后
目前主流的是使用openresty(nginx + lua)來(lái)搭建圖片處理服務(wù),使用Java的話性能可能會(huì)比較差。因?yàn)閷?duì)Java技術(shù)棧比較熟悉,前期會(huì)先使用Java實(shí)現(xiàn)。
本文的demo版本已經(jīng)上傳到github上,感興趣的小伙伴可以去看下: github.com/Shanbw/Grap…
以上就是Springboot集成GraphicsMagick的詳細(xì)內(nèi)容,更多關(guān)于Springboot集成GraphicsMagick的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- php下嘗試使用GraphicsMagick的縮略圖功能
- 詳解SpringBoot配置連接池
- springboot如何讀取配置文件(application.yml)中的屬性值
- SpringBoot定時(shí)任務(wù)兩種(Spring Schedule 與 Quartz 整合 )實(shí)現(xiàn)方法
- SpringBoot入坑筆記之spring-boot-starter-web 配置文件的使用
- 詳解springboot-修改內(nèi)置tomcat版本
- 詳解SpringBoot文件上傳下載和多文件上傳(圖文)
- SpringBoot+Maven 多模塊項(xiàng)目的構(gòu)建、運(yùn)行、打包實(shí)戰(zhàn)
相關(guān)文章
SpringMVC MVC架構(gòu)與Servlet使用詳解
MVC設(shè)計(jì)模式一般指 MVC 框架,M(Model)指數(shù)據(jù)模型層,V(View)指視圖層,C(Controller)指控制層。使用 MVC 的目的是將 M 和 V 的實(shí)現(xiàn)代碼分離,使同一個(gè)程序可以有不同的表現(xiàn)形式。其中,View 的定義比較清晰,就是用戶界面2022-10-10Spring MVC InitBinder驗(yàn)證方法
這篇文章主要介紹了Spring MVC InitBinder驗(yàn)證方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-03-03maven項(xiàng)目下solr和spring的整合配置詳解
這篇文章主要介紹了maven項(xiàng)目下solr和spring的整合配置詳解,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-11-11SpringBoot引入Thymeleaf的實(shí)現(xiàn)方法
這篇文章主要介紹了SpringBoot引入Thymeleaf的實(shí)現(xiàn)方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04關(guān)于Scanner中nextInt()、nextLine()等方法總結(jié)與問(wèn)題解決
這篇文章主要介紹了關(guān)于Scanner中nextInt()、nextLine()等方法總結(jié)與問(wèn)題解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。2022-11-11java程序員必須知道的4個(gè)書(shū)寫(xiě)代碼技巧
本篇文章主要給大家講述了作為JAVA程序員如何能寫(xiě)出高效的代碼以及運(yùn)行效率更高的代碼,一起學(xué)習(xí)分享下吧。2017-12-12解決IDEA 啟動(dòng)Tomcat控制臺(tái)亂碼問(wèn)題
今天在Idea中用Tomcat跑一個(gè)Web項(xiàng)目,啟動(dòng)后,Tomcat日志在控制臺(tái)打印出來(lái)都是亂碼,初次遇到這個(gè)問(wèn)題真的很棘手,今天小編帶領(lǐng)大家一起看看解決方法2018-07-07基礎(chǔ)不牢,地動(dòng)山搖,Java基礎(chǔ)速來(lái)刷刷
基礎(chǔ)不牢,地動(dòng)山搖,快來(lái)一起學(xué)習(xí)一下基礎(chǔ)吧,不斷地學(xué)習(xí)就算是基礎(chǔ)也會(huì)有新的認(rèn)知和收獲,加油2021-08-08