Springboot集成GraphicsMagick

 更新时间:2021年05月25日 09:21:35   作者:码bug的小砖家  
本文主要是教大家如何将GraphicsMagick命令行工具集成到Springboot项目中,便可以使用Java进行图片处理相关开发。

以什么方式集成?

JNI / 命令行(im4java)

在im4java官网中提到:

翻译过来就是: 从Java内部使用JNI运行本机代码始终会带来其他风险,对于长时间运行的进程(通常是Web应用程序服务器)尤其危险。内存损坏或分段错误(可能由故意操纵的图像触发)可能会使整个服务器瘫痪。

所以我们选择使用命令行的方式进行调用。

项目集成

1、将gm命令行工具引入到项目中

在SpringBoot集成Linux可执行命令的时候,我们将可执行文件放在了项目的resource目录下:

这里需要有一步操作就是将文件复制到宿主机:

    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")) {
            // 初始化容器的环境
            initPodEnv();
            gmPath = "gm/linux/gm";
        } else {
            throw new RuntimeException("非法操作系统:"+osName);
        }
        InputStream fisInJar = new ClassPathResource(gmPath).getInputStream();
        File file = File.createTempFile("GraphicsMagick", "_gm");
        file.setExecutable(true);
        GM_PATH = file.getAbsolutePath();
        //将jar包里的gm复制到操作系统的目录里
        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、在项目启动的时候自动初始化环境

下面只对Linux进行了自动化环境安装,mac环境主要是本地开发,自己安装环境即可:

    /**
     * 初始化容器的环境
     *
     * 安装gm所依赖的库
     */
    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");
        // 打水印时缺少依赖
        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进程池化

想象下,如果在每次进行图片处理都去 fork gm子进程,不仅代价大,而且在高并发情况下,容易造成子进程过多,导致系统负载飙高,上下文切换频繁。

所以将 gm进程 池化是很有必要的。

前提: gm提供batch批量模式,运行在此模式下的gm进程,会一直读取标准输入,逐行接收命令实时进行处理。

池化思路: 预先 fork 一批 gm 子进程,每次要运行命令时,从子进程池中挑选一个子进程,进行图片处理,处理完毕后归还连接。

具体架构:

/**
 * GM 进程池参数
 */
@ConfigurationProperties(prefix = "gm.pool")
@Data
public class GMPoolProperties {

    /**
     * 连接池最大活跃数
     */
    private int maxActive = 4;

    /**
     * 连接池最大空闲连接数
     */
    private int maxIdle = 4;

    /**
     * 连接池最小空闲连接数
     */
    private int minIdle = 2;

    /**
     * 资源池中资源最小空闲时间(单位为毫秒),达到此值后空闲资源将被移
     */
    private long minEvictableIdleTimeMillis = 300000L;

    /**
     * 连接池连接用尽后执行的动作
     */
    private WhenExhaustedAction whenExhaustedAction = WhenExhaustedAction.BLOCK;

    /**
     * 连接池没有对象返回时,最大等待时间(毫秒)
     */
    private long maxWait = 5000;

    /**
     * 定时对线程池中空闲的链接进行校验
     */
    private boolean testWhileIdle = false;

    /**
     * 空闲资源的检测周期(单位为毫秒)
     */
    private long timeBetweenEvictionRunsMillis = 10000L;

}

性能初测

1、单线程测试: 单线程循环100次

技术 耗时 平均耗时
GraphicsMagick + im4java 2110 ms 21 ms
GraphicsMagick + im4java + 池化技术 1478 ms 15 ms

总结:性能提升约29%  
2、多线程并发测试: 并发100个线程请求

技术 耗时 平均耗时
GraphicsMagick + im4java 37901 ms 379 ms
GraphicsMagick + im4java + 池化技术 22456 ms 224 ms

总结:性能提升约41%  

写在最后

目前主流的是使用openresty(nginx + lua)来搭建图片处理服务,使用Java的话性能可能会比较差。因为对Java技术栈比较熟悉,前期会先使用Java实现。

本文的demo版本已经上传到github上,感兴趣的小伙伴可以去看下: github.com/Shanbw/Grap…

以上就是Springboot集成GraphicsMagick的详细内容,更多关于Springboot集成GraphicsMagick的资料请关注脚本之家其它相关文章!

相关文章

  • 详解RabbitMQ延迟队列的基本使用和优化

    详解RabbitMQ延迟队列的基本使用和优化

    这篇文章主要介绍了详解RabbitMQ延迟队列的基本使用和优化,延迟队列中的元素都是带有时间属性的。延迟队列就是用来存放需要在指定时间被处理的元素的队列,需要的朋友可以参考下
    2023-05-05
  • Java随机数的5种获得方法(非常详细!)

    Java随机数的5种获得方法(非常详细!)

    这篇文章主要给大家介绍了关于Java随机数的5种获得方法,在实际开发中产生随机数的使用是很普遍的,所以在程序中进行产生随机数操作很重要,文中通过图文介绍的非常详细,需要的朋友可以参考下
    2023-10-10
  • java中的文件操作总结(干货)

    java中的文件操作总结(干货)

    本篇文章主要介绍了java中的文件操作总结(干货),主要有文件读写,遍历文件夹,文件夹操作等,具有一定的参考价值,感兴趣的小伙伴们可以参考一下。
    2017-02-02
  • 在SpringBoot中使用lombok的注意事项

    在SpringBoot中使用lombok的注意事项

    这篇文章主要介绍了在SpringBoot中使用lombok的注意事项,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-12-12
  • SpringBoot整合Freemarker实现页面静态化的详细步骤

    SpringBoot整合Freemarker实现页面静态化的详细步骤

    这篇文章主要介绍了SpringBoot整合Freemarker实现页面静态化,第一步要创建项目添加依赖,本文分步骤给大家详细讲解,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-10-10
  • SpringAOP 设置注入的实现步骤

    SpringAOP 设置注入的实现步骤

    这篇文章主要介绍了SpringAOP 设置注入的实现步骤,帮助大家更好的理解和学习使用Spring框架,感兴趣的朋友可以了解下
    2021-05-05
  • Java参数传递实现代码及过程图解

    Java参数传递实现代码及过程图解

    这篇文章主要介绍了Java参数传递实现代码及过程图解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-11-11
  • Java基础类库之StringBuffer类用法详解

    Java基础类库之StringBuffer类用法详解

    String类是在所有开发项目开发之中一定会使用的一个功能类。虽然String类很好用,但也有弊端——内容不允许频繁修改,所以为了解决问题,我们提供了StringBuffer类。本文就来讲讲StringBuffer类的用法
    2022-07-07
  • spring NamedContextFactory实现服务隔离的示例详解

    spring NamedContextFactory实现服务隔离的示例详解

    假设我们有个场景,我们需要实现服务之间的数据隔离、配置隔离、依赖的spring bean之间隔离,大家会有什么实现思路?今天给大家介绍spring-cloud-context里面有个NamedContextFactory可以达到上面的效果,需要的朋友可以参考下
    2024-05-05
  • mybatis教程之resultmap_动力节点Java学院整理

    mybatis教程之resultmap_动力节点Java学院整理

    这篇文章主要介绍了mybatis教程之resultmap,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-09-09

最新评论