如何构建具有捆绑依赖项的静态库 - CMake

问题描述 投票:0回答:2

我目前正在使用 CMake 创建一个静态库,该库利用 OpenCV 4 中的一些静态库( core imgcodecs video highgui imgproc )。我的目的是能够将所有必需的 OpenCV 静态库捆绑到我自己的库中,以便我可以将其作为一个库进行分发。此外,我希望我的库的用户不必在其系统上安装 OpenCV 4(但不介意用户是否必须使用 apt-get install 进行简单安装)。我知道有一些用于捆绑静态库的工具(例如使用 ar for linux)。 然而,我真正遇到的问题是 OpenCV 的所有依赖项(例如 libjpeg、libpng 等)。我不一定介意这些库是否与我的库捆绑在一起或动态链接,因为它们相对容易安装(可以使用 sudo apt-get install 安装,而 opencv4 需要从源代码构建)。

执行此操作的最佳方法是什么? 这是我当前的 CMakeLists.txt 它目前正在工作,但那是因为我正在使用 find_package(OpenCV REQUIRED) (这违背了我想要做的目的)。当我删除该行时,链接器抱怨无法找到 OpenCV 依赖项。

cmake_minimum_required(VERSION 2.8)
project(myproject)

set(CMAKE_CXX_STANDARD 14)
include_directories(${CMAKE_CURRENT_LIST_DIR}/include)
link_directories(${CMAKE_CURRENT_LIST_DIR}/lib)

find_package(OpenMP REQUIRED)
find_package(OpenCV REQUIRED)

set(JSON_BuildTests OFF CACHE INTERNAL "")
add_subdirectory(nlohmann_json)

list(APPEND LINKER_LIBS opencv_core opencv_highgui opencv_video opencv_imgcodecs libmxnet.so libncnn.a nlohmann_json::nlohmann_json)
file(GLOB SRC${CMAKE_CURRENT_LIST_DIR}/src/*.cpp${CMAKE_CURRENT_LIST_DIR}/main.cpp)

add_library(myproject ${SRC})
target_link_libraries(myproject ${LINKER_LIBS} ${OpenMP_CXX_FLAGS})

详细说明我的问题。我构建了生成 libmyproject.a 的项目。然后我获取这个库,最终从 OpenCV 库中提取符号(libopencv_core.a libopencv_highgui.a libopencv_imgcodecs.a libopencv_video.a)并将它们添加到我的库中(暂时,我还没有完成这一步,这这就是为什么在下面的示例中我链接 libopencv_*)。然后,我在一个新项目中使用我的库,其 CMakeLists.txt 如下所示:

cmake_minimum_required(VERSION 2.8)
project(myproject-driver)

set(CMAKE_CXX_STANDARD 14)
include_directories(${CMAKE_CURRENT_LIST_DIR}/include)
link_directories(${CMAKE_CURRENT_LIST_DIR}/lib)
find_package(OpenMP REQUIRED)

add_executable(myproject-driver main.cpp)
target_link_libraries(myproject-driver myproject libncnn.a ${OpenMP_CXX_FLAGS} libmxnet.so libopencv_core.a  libopencv_highgui.a  libopencv_imgcodecs.a  libopencv_video.a)

构建它会产生以下错误:

Linking CXX executable myproject-driver
/usr/bin/ld: /home/nchafni/Cyrus/myproject/lib/libopencv_imgcodecs.a(grfmt_jpeg.cpp.o): undefined reference to symbol 'jpeg_default_qtables@@LIBJPEG_8.0'
//usr/lib/x86_64-linux-gnu/libjpeg.so.8: error adding symbols: DSO missing from command line
collect2: error: ld returned 1 exit status

我该如何解决这个问题。是否有一些 CMake 命令可以为我链接所有这些依赖项?我是否需要手动跟踪这些 libopencv_* 库的每个依赖项并手动链接它们?再次,这是假设使用 libmyproject.a 的人无法使用

find_package(OpenCV REQUIRED)
,因为它不会被定义,因为他们没有在他们的机器上安装 OpenCV。

c++ opencv cmake static-libraries
2个回答
2
投票

首先,不要使用

2.8
的超级旧且过时的版本
CMake

CMake 3.x
功能更强大且使用起来非常简单。
有关现代 CMake的一些技巧。

  • 不要使用
    file(GLOB)
    ,请参阅此处为什么会这样。
  • 不要使用目录范围指令,而是使用目标指令,例如
    target_include_directories
    include_directories
  • 不要像${<PACKAGE_NAME>_LIBRARIES}那样使用
    字符串变量
    ,而是使用目标,例如
    <Package_NAME>::lib
  • 当使用targets而不是字符串变量时,该目标的所有属性(包括LINK_INTERFACE)将在调用
    target_link_libraries
    时填充到库/可执行文件中,因此不再需要
    include_directories
    link_directories 
    等等

我的项目

cmake_minimum_required(VERSION 3.14)

project(myproject)

set(CMAKE_CXX_STANDARD 14)

find_package(OpenMP REQUIRED)
find_package(OpenCV REQUIRED)

set(JSON_BuildTests OFF CACHE INTERNAL "")
add_subdirectory(nlohmann_json)

set(SOURCES ...) # list all the source files here

add_library(myproject ${SOURCES})
target_include_directories(myproject PUBLIC # give it a scope
                           ${CMAKE_CURRENT_LIST_DIR}/include
)
target_link_libraries(myproject PUBLIC # give it a scope
                      opencv_core # using the target, you will get all LINK_LIBRARIES
                      opencv_highgui 
                      opencv_video 
                      opencv_imgcodecs 
                      libmxnet.so  # where is this coming from?
                      libncnn.a # where is this coming from?
                      nlohmann_json::nlohmann_json 
                      OpenMP::OpenMP_CXX ## linking against a target, CXX_FLAGS will be populated automatically
)

我的项目驱动程序

cmake_minimum_required(VERSION 3.14)
project(myproject-driver)

set(CMAKE_CXX_STANDARD 14)

add_executable(myproject-driver main.cpp)
target_link_libraries(myproject-driver PUBLIC # give it a scope
                      myproject # gets all dependencies through the LINK_INTERFACE
)

0
投票

来自 CMake:在静态库中包含库依赖项 :

project(test)

add_library(lib1 STATIC test1.c)
add_library(lib2 STATIC test2.c)

add_custom_target(combined ALL
    COMMAND ${CMAKE_AR} rc libcombined.a $<TARGET_FILE:lib1> $<TARGET_FILE:lib2>
)
© www.soinside.com 2019 - 2024. All rights reserved.