0

0

在SpringBoot项目中如何确保FFmpeg和OpenCV的so依赖在打包后正确加载?

霞舞

霞舞

发布时间:2025-04-23 22:24:14

|

801人浏览过

|

来源于php中文网

原创

在springboot项目中如何确保ffmpeg和opencv的so依赖在打包后正确加载?

在SpringBoot项目中如何确保FFmpeg和OpenCV的so依赖在打包后正确加载?

在开发过程中,经常会遇到将FFmpeg和OpenCV集成到SpringBoot项目中的情况,尤其是当项目需要处理视频和图像时。然而,开发环境和生产环境之间的差异常常导致库文件加载问题。本文将详细探讨如何在SpringBoot项目中正确加载FFmpeg和OpenCV的so依赖,并确保打包后的应用也能正常运行。

问题背景

在使用Java 21和SpringBoot 3开发的项目中,我们依赖于FFmpeg 7.1-1.5.11和OpenCV 4.10.0-1.5.11这两个库。项目的pom.xml配置如下:


  1.5.11
  4.10.0-1.5.11
  7.1-1.5.11
  linux-x86_64
  windows-x86_64
  ${javacpp.platform.linux-x86_64}



  
    org.bytedeco
    javacpp
    ${bytedeco.version}
  
  
    org.bytedeco
    javacv
    ${bytedeco.version}
  
  
    org.bytedeco
    opencv
    ${opencv-platform.version}
    ${javacpp.platform}
  
  
    org.bytedeco
    ffmpeg
    ${ffmpeg-platform.version}
    ${javacpp.platform}
  

当项目打包成jar后,FFmpeg和OpenCV的so依赖应该被包含在jar包中的BOOT-INF/lib目录下。然而,在运行时,这些依赖并没有被正确加载,导致如下错误:

Caused by: java.lang.ExceptionInInitializerError: Exception java.lang.UnsatisfiedLinkError: no jniavutil in java.library.path: /usr/java/packages/lib:/usr/lib64:/lib64:/lib:/usr/lib [in thread "http-nio-8100-exec-1"]

在开发环境中,IDE会自动将依赖添加到classpath中,因此可以正常加载。但在生产环境中,jar包的运行方式不同,导致这些依赖无法被找到。

解决方案

为了解决这个问题,我们需要在SpringBoot应用中编程方式配置JavaCPP,以便在应用启动时正确加载这些本地库。

  1. 在应用启动时配置JavaCPP

    文心快码
    文心快码

    文心快码(Comate)是百度推出的一款AI辅助编程工具

    下载

    我们可以创建一个配置类,在应用启动时执行必要的配置:

    package com.demo.config;
    
    import org.bytedeco.javacpp.Loader;
    import org.springframework.context.annotation.Configuration;
    import javax.annotation.PostConstruct;
    
    @Configuration
    public class JavaCppConfig {
    
        @PostConstruct
        public void init() {
            // 设置JavaCPP提取本地库的目录
            System.setProperty("org.bytedeco.javacpp.extract", "true");
            System.setProperty("org.bytedeco.javacpp.extractDir", "/tmp/javacpp-native-libs");
            
            // 预加载需要的库,这会触发JavaCPP从jar中提取库文件
            try {
                // 预加载FFmpeg相关库
                Loader.load(org.bytedeco.ffmpeg.global.avutil.class);
                Loader.load(org.bytedeco.ffmpeg.global.avcodec.class);
                Loader.load(org.bytedeco.ffmpeg.global.avformat.class);
                Loader.load(org.bytedeco.ffmpeg.global.swscale.class);
                
                // 预加载OpenCV库
                Loader.load(org.bytedeco.opencv.global.opencv_core.class);
                Loader.load(org.bytedeco.opencv.global.opencv_imgproc.class);
                
                System.out.println("Native libraries loaded successfully");
            } catch (Exception e) {
                System.err.println("Failed to load native libraries: "   e.getMessage());
                e.printStackTrace();
            }
        }
    }
  2. 自定义Docker镜像构建

    如果项目需要构建成Docker镜像,我们可以创建一个自定义的Dockerfile来预先提取这些本地库并配置好路径:

    FROM openjdk:21-slim
    
    # 安装程序运行所需的库
    RUN apt-get update && apt-get install -y \
        libgomp1 \
        && rm -rf /var/lib/apt/lists/*
    
    # 创建目录用于提取本地库
    RUN mkdir -p /opt/javacpp-native-libs
    ENV JAVACPP_EXTRACT_DIR=/opt/javacpp-native-libs
    
    # 设置工作目录
    WORKDIR /app
    
    # 复制应用jar包
    COPY target/*.jar app.jar
    
    # 预先提取所有本地库到指定目录
    RUN mkdir -p /tmp/extract && \
        cd /tmp/extract && \
        java -Dorg.bytedeco.javacpp.extract=true \
             -Dorg.bytedeco.javacpp.extractDir=/opt/javacpp-native-libs \
             -jar /app/app.jar --extract-native-libraries && \
        rm -rf /tmp/extract
    
    # 设置系统属性,指向提取的本地库位置
    ENV JAVA_OPTS="-Dorg.bytedeco.javacpp.extractDir=/opt/javacpp-native-libs -Dorg.bytedeco.javacpp.extract=false"
    
    # 启动应用
    ENTRYPOINT ["sh", "-c", "java $JAVA_OPTS -jar app.jar"]

通过上述方法,我们可以在应用启动时正确加载FFmpeg和OpenCV的so依赖,并且确保在Docker容器中也能正常运行,而无需修改基础镜像或将so文件添加到外部目录。

注意事项

  • JavaCPP需要能够写入临时目录来提取本地库。
  • 在Docker环境中,可能需要添加-Dorg.bytedeco.javacpp.maxphysicalbytes=0来防止内存限制问题。
  • 如果项目需要在多个平台(如Windows和Linux)上运行,需注意配置正确的平台属性。

通过在启动应用时添加以下系统属性,也可以尝试解决问题:

java -Djava.library.path=path/to/extracted/libs -Djavacpp.platform=linux-x86_64 -jar your-app.jar

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

825

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

724

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

728

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

395

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

445

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

428

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16861

2023.08.03

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

7

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 6.3万人学习

Git 教程
Git 教程

共21课时 | 2.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号