CMakeList.txt在大型文件应用(SLAM常用库添加依赖项)

2024-03-08 09:20

本文主要是介绍CMakeList.txt在大型文件应用(SLAM常用库添加依赖项),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

摘自:https://zhuanlan.zhihu.com/p/149191302?utm_source=wechat_session

CMakeList.txt在大型文件应用(SLAM常用库添加依赖项)

晓伟Liu

晓伟Liu

CSDN无人驾驶博客ID try_again_later

基本的CMakeLists.txt并不难,主要有生成库、生成执行文件、链接二者以及找库find_package、找头文件include_directories、生成执行文件add_executable、链接库和执行文件target_link_libraries两种方法组成。

但是遇到大型库的编写,目标是生成一个新的大型库myslam,生成执行文件、链接二者。需要提前声明生成执行文件在bin,库在lib中,并把OPenCV、Sophus这些库整合到一个大型的第三方库THIRD_PARTY_LIBS中,供后面的target_link_libraries(myslam ${THIRD_PARTY_LIBS})用。myslam中包括两部分:frame.cpp、mappoint.cpp、map.cpp等+第三方库THIRD_PARTY_LIBS


一、简单的

1)从头开始:(生成库、生成执行文件、链接二者)

CMakeLists.txt

add_library( hello_shared SHARED libHelloSLAM.cpp ) # 生成库

add_executable( useHello useHello.cpp ) #执行文件

target_link_libraries( useHello hello_shared ) # 链接库到执行文件上

1、先生成库文件libhello_shred.so,需要libHelloSLAM.cpp和add_library(hello_shared SHARED libHelloSLAM.cpp)

2、之后要建立头文件libhellSLAM.h告诉如何调用库里的函数,需要把库连接到可执行文件名中 target_link_libraried(useHello hello_shared)

3、主函数useHello.cpp使用头文件

2)利用现成的库:

只需要建立头文件,说明怎么使用即可。

3)大型的库只需修改CMakeList.txt即可(找库、添加头文件、生成执行文件、链接库和执行文件

头文件也已经建立好了,只需找到库然后加入头文件即可使用

CMakeLists.txt

find_package( OpenCV REQUIRED ) #寻找OpenCV库
include_directories( ${OpenCV_INCLUDE_DIRS})  #添加头文件

add_executable( imageBasics imageBasics.cpp)    #添加可执行文件
target_link_libraries( imageBasics ${OpenCV_LIBS} ) #链接OpenCV库

 


二、大型文件

 

1、我们的目标是:

写一个VO库myslam库,这个库需要自己写在include里的.cpp 五大类以及一些第三方库(Eigen、OpenCV、Sophus、G2O)。

之后需要测试程序生成执行文件run_vo,最后链接myslam库的run_vo,跑效果。

add_library( myslam SHARED

frame.cpp mappoint.cpp map.cpp camera.cpp config.cpp

g2o_types.cpp visual_odometry.cpp )

add_executable( run_vo run_vo.cpp )

target_link_libraries( run_vo myslam )

2、步骤:

根目录的CMakeList.txt生成新的第三方库THIRD_PARTY_LIBS,需要include所有第三方库头文件,并set路径参数。

很多第三方库文件则被整合成了一个大的THIRD_PARTY_LIBS路径变量,以备后面src中生成myslam用target_link_libraries.

src中的CMakeList.txt中生成myslam库,需要五大类.cpp链接到上层定义的THIRD_PARTY_LIBS库文件路径。

test中的CMakeList.txt中就是简单的生成可执行文件run_vo,并将源文件链接到我们写的myslam库文件上。

接下来,对这三个CMakeList.txt分别进行介绍:

先来看根目录下的这个CMakeList.txt:

#定义需求版本和工程名称#
cmake_minimum_required(VERSION 2.8)
project (myslam)#cmake相关的一些设定#
set( CMAKE_CXX_COMPILER "g++" )
set( CMAKE_BUILD_TYPE "Release" )
set( CMAKE_CXX_FLAGS "-std=c++11 -march=native -O3" )#添加cmake_modules到CMAKE_MODULE_PATH路径列表中和
list( APPEND CMAKE_MODULE_PATH ${PROJECT_SOURCE_DIR}/cmake_modules )
#设定可执行文件与库文件的输出路径bin、lib
set( EXECUTABLE_OUTPUT_PATH ${PROJECT_SOURCE_DIR}/bin )
set( LIBRARY_OUTPUT_PATH ${PROJECT_SOURCE_DIR}/lib )############### 找第三方库和添加头文件 ######################
# Eigen
include_directories( "/usr/include/eigen3" )
# OpenCV
find_package( OpenCV 3.1 REQUIRED )
include_directories( ${OpenCV_INCLUDE_DIRS} )
# Sophus 
find_package( Sophus REQUIRED )
include_directories( ${Sophus_INCLUDE_DIRS} )
# G2O
find_package( G2O REQUIRED )
include_directories( ${G2O_INCLUDE_DIRS} )#根据库文件路径参数生成第三方库THIRD_PARTY_LIBS
set( THIRD_PARTY_LIBS ${OpenCV_LIBS}${Sophus_LIBRARIES}g2o_core g2o_stuff g2o_types_sba
)
############### 自己写的头文件 ######################
include_directories( ${PROJECT_SOURCE_DIR}/include )#增加子目录#
add_subdirectory( src )
add_subdirectory( test )

然后src中的CMakeList.txt:

#由此一堆生成自己写的一个库文件,名为myslam,这个库是链接在第三方库基础上的。
add_library( myslam SHAREDframe.cppmappoint.cppmap.cppcamera.cppconfig.cppg2o_types.cppvisual_odometry.cpp
)# myslam库需要链接上方定义好的第三方库,${THIRD_PARTY_LIBS}路径参数。
target_link_libraries( myslam${THIRD_PARTY_LIBS}
)

然后test中的CMakeList.txt:

 #最终到这里,添加可执行文件
add_executable( run_vo run_vo.cpp )
#可执行文件链接在自己写的库文件myslam上#
target_link_libraries( run_vo myslam )

注解:

1、当前目录名称为 {PROJECT_SOURCE_DIR},路径表达为 ${PROJECT_SOURCE_DIR}

2、list( APPEND CMAKE_MODULE_PATH ${PROJECT_SOURCE_DIR}/cmake_modules )这句的意思也就很明了了,就是将工程根目录下的cmake_modules文件夹路径,添加到CMAKE_MODULE_PATH路径列表中。

3、add_subdirectory 命令。我们的项目各个子项目都在一个总的项目根目录下,该命令可以将指定的文件夹加到build任务列表中。


三、解析多传感器融合的CMakeLists.txt

cmake_minimum_required(VERSION 2.8.3)
project(lidar_localization)SET(CMAKE_BUILD_TYPE "Release")
SET(CMAKE_CXX_FLAGS_RELEASE "$ENV{CXXFLAGS} -O3 -Wall")
add_compile_options(-std=c++11)
add_definitions(-std=c++11)# 加载catkin宏和指定对其他ROS功能包的依赖关系
find_package(catkin REQUIRED COMPONENTSroscpprospystd_msgspcl_rosgeometry_msgstfeigen_conversions
)# 把所有库列为一个集合,后面被target_link_libraries用,避免库名字太长
set(ALL_TARGET_LIBRARIES "")# cmake包放在一个文件夹下
include(cmake/glog.cmake)
include(cmake/PCL.cmake)
include(cmake/eigen.cmake)
include(cmake/geographic.cmake)
include_directories(include ${catkin_INCLUDE_DIRS})# 指定头文件路径
include(cmake/global_defination.cmake)# 空的?
catkin_package()# 所有cpp合并到一个集合(多个node文件编到一个可执行文件中会出错,踢出去)
file(GLOB_RECURSE ALL_SRCS "*.cpp")file(GLOB_RECURSE NODE_SRCS "src/*_node.cpp")
file(GLOB_RECURSE THIRD_PARTY_SRCS "third_party/*.cpp")
list(REMOVE_ITEM ALL_SRCS ${NODE_SRCS})
list(REMOVE_ITEM ALL_SRCS ${THIRD_PARTY_SRCS})# 生成可执行文件,链接库
add_executable(test_frame_node src/test_frame_node.cpp ${ALL_SRCS})
target_link_libraries(test_frame_node ${catkin_LIBRARIES} ${ALL_TARGET_LIBRARIES})

1、find_package和catkin_package

find_package 是 cmake 中常见的宏,用于加载 catkin 宏和指定对其他 ROS 功能包的依赖关系。构建此功能包A需要依赖于

roscpp
rospy
std_msgs
pcl_ros
geometry_msgs
tf
eigen_conversions

catkin_package 宏是 catkin 的宏之一,声明要传递给依赖项目的内容,生成 cmake 配置文件。对依赖于功能包A的其他功能包B来需要依赖其他其他。。。

2、set,file

set(ALL_TARGET_LIBRARIES "")

把所有库列为一个集合,后面被target_link_libraries用,避免库名字太长

与之对应的是:

file(GLOB_RECURSE ALL_SRCS "*.cpp")

所有cpp合并到一个集合

有个类似的表达方式为:将所有.cpp源文件赋值给参数 DIR_SRCS

AUX_SOURCE_DIRECTORY(. DIR_SRCS)

在最后生成可执行文件、链接库的时候,只需要用到ALL_SRCSALL_TARGET_LIBRARIES即可。

# 生成可执行文件,链接库
add_executable(test_frame_node src/test_frame_node.cpp ${ALL_SRCS})
target_link_libraries(test_frame_node ${catkin_LIBRARIES} ${ALL_TARGET_LIBRARIES})

3、.cmake包

# cmake包放在一个文件夹下
include(cmake/glog.cmake)
include(cmake/PCL.cmake)
include(cmake/eigen.cmake)
include(cmake/geographic.cmake)
include_directories(include ${catkin_INCLUDE_DIRS})# 指定头文件路径
include(cmake/global_defination.cmake)v

四、SLAM中常用库的CMakeList.txt写法(如何添加各种依赖库)

1.添加Eigen头文件

Eigen通过apt-get安装之后,我们要怎么使用这个依赖库呢?Eigen有一点很奇怪,Eigen库只有头文件,所以我们在CMakeLists.txt中是不需要写target_link_libraries,因为我们是通过apt安装的,所以很容易的知道Eigen库的位置就是在/usr/include/eigen3

#添加头文件
include_directories("/usr/include/eigen3")

因为我们已经知道Eigen具体的位置,就不用使用find_package来寻找了,虽然有些粗暴,但是简单有效。

2.添加Pangolin依赖
Pangolin的安装也很简单,功能主要就是做三维的可视化显示,主要依赖库是OpenGL,通过apt也很好安装。

find_package(Pangolin)include_directories(${Pangolin_INCLUDE_DIRS})add_executable(project_name filename.cpp)target_link_libraries(project_name ${Pangolin_LIBRARIES})

3.添加Sophus依赖

Sophus实际上是Eigen库的扩展模块,Eigen中虽然有几何模块,但是没有提供李代数的支持,所以Sophus算是一个比较好的李代数库,安装参考之前博文。

find_package(Sophus REQUIRED)
include_directories(${Sophus_INCLUDE_DIRS})
add_executable(project_name project_name.cpp)
target_link_libraries(project_name ${Sophus_LIBRARIES})

4.添加OpenCV依赖

OpenCV经常会出现版本不兼容的问题,LZ同时安装了OpenCV2和OpenCV3两个版本,所以在CMakeLists.txt要指定OpenCV的版本。

#指定OpenCV的版本是3.1
find_package(OpenCV 3.1 REQUIRED)include_directories(${OpenCV_INCLUDE_DIRS})add_executable(project_name filename.cpp)target_link_libraries(project_name ${OpenCV_LIBS})

添加OpenCV要注意一个问题,大小写!很重要!大小写!

5.添加PCL依赖

这个点云库在SLAM应用中还是蛮重要的,通常情况下好像也都有安装,安装很简单,这里还是讲一下CMakeLists.txt中怎么写:

find_package(PCL REQUIRED COMPONENT common io)include_directories(${PCL_INCLUDE_DIRS})add_definitions(${PCL_DEFINITIONS})target_link_libraries(project_name ${PCL_LIBRARIES})

6.添加Ceres依赖

Ceres是Google出品的一个优化库,安装编译都在LZ之前写过一个SLAM安装大全里都有。因为Ceres不是常用的库,所以需要添加一个cmake_modules。

#这行代码就是添加查找Ceres的一个文件
list(APPEND CMAKE_MODULE_PATH ${PROJECT_SOURCE_DIR}/cmake_modules)find_package(Ceres REQUIRED)include_directories(${CERES_INCLUDE_DIRS})target_link_libraries(project_name ${CERES_LIBRARIES})

7添加G2O的依赖

其实这个也是千篇一律,换汤不换药。这里也是要注意几个问题,第一个,要在cmake_module中假如findG2O的文件。第二个,注意大小写问题,还有数字0和字母0,这个还是要注意的。

list(APPEND CMAKE_MODULE_PATH ${PROJECT_SOURCE_DIR}/cmake_modules)find_package(G2O REQUIRED)include_directories(${G20_INCLUDE_DIRS})target_link_libraries(project_name g2o_core g2o_stuff)

 

编辑于 2020-06-18

这篇关于CMakeList.txt在大型文件应用(SLAM常用库添加依赖项)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/786646

相关文章

在Ubuntu上部署SpringBoot应用的操作步骤

《在Ubuntu上部署SpringBoot应用的操作步骤》随着云计算和容器化技术的普及,Linux服务器已成为部署Web应用程序的主流平台之一,Java作为一种跨平台的编程语言,具有广泛的应用场景,本... 目录一、部署准备二、安装 Java 环境1. 安装 JDK2. 验证 Java 安装三、安装 mys

Python中构建终端应用界面利器Blessed模块的使用

《Python中构建终端应用界面利器Blessed模块的使用》Blessed库作为一个轻量级且功能强大的解决方案,开始在开发者中赢得口碑,今天,我们就一起来探索一下它是如何让终端UI开发变得轻松而高... 目录一、安装与配置:简单、快速、无障碍二、基本功能:从彩色文本到动态交互1. 显示基本内容2. 创建链

Node.js 中 http 模块的深度剖析与实战应用小结

《Node.js中http模块的深度剖析与实战应用小结》本文详细介绍了Node.js中的http模块,从创建HTTP服务器、处理请求与响应,到获取请求参数,每个环节都通过代码示例进行解析,旨在帮... 目录Node.js 中 http 模块的深度剖析与实战应用一、引言二、创建 HTTP 服务器:基石搭建(一

Java 枚举的常用技巧汇总

《Java枚举的常用技巧汇总》在Java中,枚举类型是一种特殊的数据类型,允许定义一组固定的常量,默认情况下,toString方法返回枚举常量的名称,本文提供了一个完整的代码示例,展示了如何在Jav... 目录一、枚举的基本概念1. 什么是枚举?2. 基本枚举示例3. 枚举的优势二、枚举的高级用法1. 枚举

java中VO PO DTO POJO BO DO对象的应用场景及使用方式

《java中VOPODTOPOJOBODO对象的应用场景及使用方式》文章介绍了Java开发中常用的几种对象类型及其应用场景,包括VO、PO、DTO、POJO、BO和DO等,并通过示例说明了它... 目录Java中VO PO DTO POJO BO DO对象的应用VO (View Object) - 视图对象

python中poetry安装依赖

《python中poetry安装依赖》本文主要介绍了Poetry工具及其在Python项目中的安装和使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随... 目录前言1. 为什么pip install poetry 会造成依赖冲突1.1 全局环境依赖混淆:1

IDEA常用插件之代码扫描SonarLint详解

《IDEA常用插件之代码扫描SonarLint详解》SonarLint是一款用于代码扫描的插件,可以帮助查找隐藏的bug,下载并安装插件后,右键点击项目并选择“Analyze”、“Analyzewit... 目录SonajavascriptrLint 查找隐藏的bug下载安装插件扫描代码查看结果总结Sona

Go信号处理如何优雅地关闭你的应用

《Go信号处理如何优雅地关闭你的应用》Go中的优雅关闭机制使得在应用程序接收到终止信号时,能够进行平滑的资源清理,通过使用context来管理goroutine的生命周期,结合signal... 目录1. 什么是信号处理?2. 如何优雅地关闭 Go 应用?3. 代码实现3.1 基本的信号捕获和优雅关闭3.2

正则表达式高级应用与性能优化记录

《正则表达式高级应用与性能优化记录》本文介绍了正则表达式的高级应用和性能优化技巧,包括文本拆分、合并、XML/HTML解析、数据分析、以及性能优化方法,通过这些技巧,可以更高效地利用正则表达式进行复杂... 目录第6章:正则表达式的高级应用6.1 模式匹配与文本处理6.1.1 文本拆分6.1.2 文本合并6

python中的与时间相关的模块应用场景分析

《python中的与时间相关的模块应用场景分析》本文介绍了Python中与时间相关的几个重要模块:`time`、`datetime`、`calendar`、`timeit`、`pytz`和`dateu... 目录1. time 模块2. datetime 模块3. calendar 模块4. timeit