一、说明
最近用cmake开发东西,编译vs时候,发现debug和release版本的lib库的依赖项问题,故此小结一下。若有不对之处,还请看官多多指教。
使用的工程有自己编写的工程,也有借用第三方库的工程,还有没有办法找到源码的,只有dll和lib库,没有区分debug和release版本的。
所以还是分开说,一种自己工程库,一种是第三方库。在写完cmake代码,生成vs后,都可以自动的添加链接库,debug和release版本泾渭分明。
二、自己工程之间的引用
先说,自己编写的工程,工程直接的相互调用,这个就不用多说了。Cmake还是要调用target_link_libraries来链接自己的想要链接的动态库。
但是需用做些设置,就可以自动的区分debug和release版本了。
首先,需实现cmake定义如下:
#这个就是定义各个版本对应的后缀,例如d,debug版本后缀,当然你想定义为其他, #自己修改这块就可以了。 SET(CMAKE_DEBUG_POSTFIX "d" CACHE STRING "add a postfix,usually d on windows") SET(CMAKE_RELEASE_POSTFIX "" CACHE STRING "add a postfix,usually empty on windows") SET(CMAKE_RELWITHDEBINFO_POSTFIX "rd" CACHE STRING "add a postfix,usually empty on windows") SET(CMAKE_MINSIZEREL_POSTFIX "s" CACHE STRING "add a postfix,usually empty on windows") # Set the build postfix extension according to what configuration is being built. IF (CMAKE_BUILD_TYPE MATCHES "Release") SET(CMAKE_BUILD_POSTFIX "${CMAKE_RELEASE_POSTFIX}") ELSEIF (CMAKE_BUILD_TYPE MATCHES "MinSizeRel") SET(CMAKE_BUILD_POSTFIX "${CMAKE_MINSIZEREL_POSTFIX}") ELSEIF(CMAKE_BUILD_TYPE MATCHES "RelWithDebInfo") SET(CMAKE_BUILD_POSTFIX "${CMAKE_RELWITHDEBINFO_POSTFIX}") ELSEIF(CMAKE_BUILD_TYPE MATCHES "Debug") SET(CMAKE_BUILD_POSTFIX "${CMAKE_DEBUG_POSTFIX}") ELSE() SET(CMAKE_BUILD_POSTFIX "") ENDIF()
以上代码我们是在外层的CMakeLists.txt中实现的。
接着下来cmake代码是在内层的cMakeLists.txt中实现。主要是使用外层定义的东西。
Cmake代码如下:
# Set the library extension according to what configuration is being built. IF(CMAKE_DEBUG_POSTFIX) SET(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -DRW_LIBRARY_POSTFIX=${CMAKE_DEBUG_POSTFIX}") ENDIF() IF(CMAKE_RELEASE_POSTFIX) SET(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} -DRW_LIBRARY_POSTFIX=${CMAKE_RELEASE_POSTFIX}") ENDIF() IF(CMAKE_RELWITHDEBINFO_POSTFIX) SET(CMAKE_CXX_FLAGS_RELWITHDEBINFO "${CMAKE_CXX_FLAGS_RELWITHDEBINFO} -DRW_LIBRARY_POSTFIX=${CMAKE_RELWITHDEBINFO_POSTFIX}") ENDIF() IF(CMAKE_MINSIZEREL_POSTFIX) SET(CMAKE_CXX_FLAGS_MINSIZEREL "${CMAKE_CXX_FLAGS_MINSIZEREL} -DRW_LIBRARY_POSTFIX=${CMAKE_MINSIZEREL_POSTFIX}") ENDIF()
其中的RW_LIBRARY_POSTFIX是我们工程相关的名称,你们可以自己设置。在工程属性的出现,如下所示:
三、第三方库
关于第三方库,情况比较复杂。有源码的,没有源码的,静态的,动态的,只有release,没有debug的。我们现在做的是要求,至少有release版本的的lib库。如没有就不会添加到所需的工程的依赖项中。
因为第三方库,在项目组中,各自的工程各自知道需用什么第三方库,所以就写了一个cmake的宏定义来实现。
下面是cmake原文代码:
#link library for debug and release by yourself. MACRO(RW_LINK_LIBRARY BASE_LIBRARY_NAME DEBUGSUFFIX EXSUFFIX) set(DEBUG_LIB ${PROJECT_SOURCE_DIR}/${BASE_LIBRARY_NAME}${DEBUGSUFFIX}) set(RELEASE_LIB ${PROJECT_SOURCE_DIR}/${BASE_LIBRARY_NAME}${EXSUFFIX}) IF(EXISTS ${RELEASE_LIB}) <span style="white-space:pre"> </span>target_link_libraries(${PROJECT_NAME} optimized ${RELEASE_LIB}) <span style="white-space:pre"> </span>IF(EXISTS ${DEBUG_LIB}) <span style="white-space:pre"> </span> target_link_libraries(${PROJECT_NAME} debug ${DEBUG_LIB}) <span style="white-space:pre"> </span>ELSE() <span style="white-space:pre"> </span> target_link_libraries(${PROJECT_NAME} debug ${RELEASE_LIB}) <span style="white-space:pre"> </span>ENDIF(EXISTS ${DEBUG_LIB}) ENDIF(EXISTS ${RELEASE_LIB}) ENDMACRO() MACRO(RW_LINK_3RD_PART_LIBRARY FULL_LIBRARY_DEBUGNAME FULL_LIBRARY_RELEASENAME) IF(EXISTS ${FULL_LIBRARY_RELEASENAME}) target_link_libraries(${PROJECT_NAME} optimized ${FULL_LIBRARY_RELEASENAME}) IF(NOT EXISTS ${FULL_LIBRARY_DEBUGNAME}) target_link_libraries(${PROJECT_NAME} debug ${FULL_LIBRARY_RELEASENAME}) ELSE() target_link_libraries(${PROJECT_NAME} debug ${FULL_LIBRARY_DEBUGNAME}) ENDIF(NOT EXISTS ${FULL_LIBRARY_DEBUGNAME}) ENDIF(EXISTS ${FULL_LIBRARY_RELEASENAME}) ENDMACRO()
宏的使用方式为:
第一个宏的使用在对应的工程下,lib库可以写相对路径。
RW_LINK_LIBRARY(free_image/FreeImage"d.lib"".lib")
第一个参数为含路径名称的lib库名称,后面分别为debug和release版本的区分。
RW_LINK_LIBRARY(free_image/FreeImage"d""")内部自动来识别andriod或ios或msvc等。
第二个宏的用法为:
RW_LINK_3RD_PART_LIBRARY(${GDAL_LIBRARY_DEBUG}${GDAL_LIBRARY})
其中里面写的是两个lib库的路径变量。
两个宏定义的原则均为,当没有debug版本的lib库时候,使用release版本的lib库。
ps:
Cmake也是边做边学,若有不足之处,还请多多包含。
其中,也有实现了功能,而不知所以的地方。
若有问题,请不吝指正。