使用具有外部依赖关系的库构建多个子项目

时间:2019-01-21 14:06:04

标签: c++ cmake itk

我正在努力处理一个包含CMake(版本3.11)和子目录的项目。它包括ITK作为外部库。

我想创建可在MyApp等不同子目录中使用的库,以构建基于MyLib的多个应用程序,而MyLib中的每个更改都会在其他应用程序中更新。 MyLib是仅标头的模板库。此外,这种库在VisualStudio中没有自己的项目。

我的问题是:

  1. 在这种情况下使用CMake的正确方法是什么?我已经搜索并尝试了一些示例,但未成功。理想情况下,我可以独立于MyLib构建应用程序。顶部的CMakeLists.txt可能不会从库中提取包含内容。这应该由应用程序完成。
  2. 如果MyLib成为非仅标题的库,将会发生什么变化?将来我可能会添加非模板化类。

奖金问题:

  1. 如何将仅标头的库添加到VisualStudio?这是为了方便起见。

感谢您的帮助,如有必要,我可以提供更多详细信息。

源环境如下:

ProjectDirectory
-  CMakeLists.txt
-- MyLib
   - CMakeLists.txt
   -- include
      - File1.h (template header)
      - File1.hxx (template body)
   -- src
      - (empty for now)
-- MyApp
   - CMakeLists.txt
   - AppFile1.cxx (File containing main function)

这是一个资源外生成项目。理想情况下,将已编译的库和应用程序放在这样的目录中(在Windows上):

└── bin
    ├── Debug
    │   ├── MyApp.exe
    │   └── MyLib.lib
    └── Release
        ├── MyApp.exe
        └── MyLib.lib

ProjectDirectory CMakeLists.txt

cmake_minimum_required(VERSION 3.10)
project(project_name)

# Setup build locations.
if(NOT CMAKE_RUNTIME_OUTPUT_DIRECTORY)
  set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}/bin)
endif()
if(NOT CMAKE_LIBRARY_OUTPUT_DIRECTORY)
  set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}/lib)
endif()
if(NOT CMAKE_ARCHIVE_OUTPUT_DIRECTORY)
  set(CMAKE_ARCHIVE_OUTPUT_DIRECTORY ${CMAKE_BINARY_DIR}/lib)
endif()

add_subdirectory(MyLib)
add_subdirectory(MyApp)

MyLib CMakeLists.txt:

find_package(ITK REQUIRED
    COMPONENTS RTK ITKImageIO)
include(${ITK_USE_FILE})

set(HDRS
    ${CMAKE_CURRENT_SOURCE_DIR}/include/File1.h
    ${CMAKE_CURRENT_SOURCE_DIR}/include/File1.hxx
)

add_library(MyLib ${HDRS})

target_include_directories(MyLib
    PUBLIC
        $<BUILD_INTERFACE:${CMAKE_CURRENT_SOURCE_DIR}/include>
        $<INSTALL_INTERFACE:include>
    PRIVATE
        src)

target_link_libraries(MyLib
    PUBLIC ${ITK_LIBRARIES})

export(TARGETS MyLib FILE MyLibConfig.cmake)

MyApp CMakeLists.txt

find_package(ITK REQUIRED
    COMPONENTS RTK ITKRegistrationMethodsv4 ITKTransformIO ITKImageIO)
include(${ITK_USE_FILE})

set(SRCS
    AppFile1.cxx)

add_executable(MyApp ${SRCS})
include_directories(${CMAKE_SOURCE_DIR}/MyLib/include)
target_link_libraries(MyApp
    ${ITK_LIBRARIES}
    MyLib)

Edit1:从MyLib和MyApp CMakeLists.txt中删除项目(X),然后将目标更改为MyLib。

1 个答案:

答案 0 :(得分:0)

您可以有一个空项目,只需创建一个自定义目标即可:

add_custom_target(MyLib SOURCES ${CMAKE_MYLIB_SRC} ${CMAKE_MYLIB_HDR})

由于您的代码仅是标头,因此任何依赖目标都将看到文​​件中的更改并重新编译。除此之外,没有任何事情。您不必创建目标,尽管第三个问题就是答案。

但是,如果ITK只是您的依赖项,那么既然您有了目标,则可以在其上添加PUBLIC属性,例如由于您的库而需要链接到它的依赖库。

在这种情况下,需要为其添加代码:

target_link_library(${NEWTARGET} PRIVATE MyLib) # PUBLIC/PRIVATE has to be tailored

这是问题1的答案。

如果您的库仅变为非标题,只需更改add_custom_target调用。