如何使用捆绑的依赖项构建静态库 - CMake

How to build static library with bundled dependencies - CMake

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

执行此操作的最佳方法是什么? 这是我目前的 CMakeLists.txt 它目前正在工作,但那是因为我正在使用 find_package(OpenCV REQUIRED) (这违背了我想要做的事情的目的)。当我删除该行时,linker 抱怨无法找到 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)并将它们添加到我的库中(暂时,我尚未完成此步骤,这就是为什么在下面的示例中我 linking 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 命令可以为我 link 所有这些依赖项?我是否需要手动跟踪那些 libopencv_* 库和 link 那些手动依赖项的每个依赖项?再一次,这是假设使用 libmyproject.a 的人不能使用 find_package(OpenCV REQUIRED) 因为它不会被定义,因为他们没有在他们的机器上安装 OpenCV。

首先,不要使用CMake的超旧版本2.8
CMake 3.x 功能更强大且使用起来非常简单。
现代 CMake.

的一些技巧
  • 不要使用 file(GLOB),查看 为什么会这样。
  • 不要使用 directory wide 指令,而是使用 target 指令,例如target_include_directories 对比 include_directories.
  • 不要像 ${<PACKAGE_NAME>_LIBRARIES} 那样使用 字符串变量 ,而是使用 目标 ,例如<Package_NAME>::lib
  • 当使用目标而不是字符串变量时,所有属性(包括[=该目标的 30=]LINK_INTERFACE) 将在调用 target_link_libraries 时填充到 library/executable,因此不再有 include_directorieslink_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
)

myprojec-driver

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
)