Ubuntu2004 CMake 使用基础

2024-04-27 13:12
文章标签 基础 使用 cmake ubuntu2004

本文主要是介绍Ubuntu2004 CMake 使用基础,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、环境安装

  1.  win10安装wsl ubuntu2004

    #windows c盘工程目录建立软链

    ln -s /mnt/c /home/vrviu/

  2. 安装cmake、c++编译工具

    apt install -y cmake g++

二、CMakeLists.txt讲解

  1. 准备工作

    首先,在/home/vrviu 目录建立一个 cmake 目录

    以后我们所有的 cmake 练习都会放在cmake 的子目录下

    然后在 cmake 建立第一个练习目录 t1

    cd cmake

    mkdir t1

    cd t1

    在 t1 目录建立 main.c 和 CMakeLists.txt(注意文件名大小写):

    main.c 文件内容:

    //main.c

    #include <stdio.h>

    int main()

    {

    printf(“Hello World from t1 Main!\n”);

    return 0;

    }

    CmakeLists.txt 文件内容:

    PROJECT (HELLO)

    SET(SRC_LIST main.c)

    MESSAGE(STATUS "This is BINARY dir " ${HELLO_BINARY_DIR})

    MESSAGE(STATUS "This is SOURCE dir "${HELLO_SOURCE_DIR})

    ADD_EXECUTABLE(hello SRC_LIST)

  2. 开始构建

    指令: cmake .
    成功建立如下:

    包括:CMakeCache.txt、CMakeFiles、cmake_install.cmake、Makefile等中间文件。
    指令:make

    PS:可以使用make VERBOSE=1来查看make构建的详细过程。
    这个时候已经生成了hello.
    指令:./hello

    以上是cmake构建的全部过程。

  3. 详细解释

     CMakeLists.txt,是cmake 的构建定义文件,文件名

    是大小写相关的,如果工程存在多个目录,需要确保每个要管理的目录都存在一个

    CMakeLists.txt。

    PROJECT 指令的语法是:

    PROJECT(projectname [CXX] [C] [Java])

    同时 cmake 系统也帮助我们预定义了 PROJECT_BINARY_DIR 和 PROJECT_SOURCE_DIR

    变量,他们的值分别跟 HELLO_BINARY_DIR 与 HELLO_SOURCE_DIR 一致。

    建议以后直接使用 PROJECT_BINARY_DIR,PROJECT_SOURCE_DIR,即

    使修改了工程名称,也不会影响这两个变量。如果使用了

    <projectname>_SOURCE_DIR,修改工程名称后,需要同时修改这些变量。

    SET 指令的语法是:

    SET(VAR [VALUE] [CACHE TYPE DOCSTRING [FORCE]])

    现阶段,你只需要了解 SET 指令可以用来显式的定义变量即可。

    比如我们用到的是 SET(SRC_LIST main.c),如果有多个源文件,也可以定义成:

    SET(SRC_LIST main.c)

    MESSAGE 指令的语法是:

    MESSAGE([SEND_ERROR | STATUS | FATAL_ERROR] "message to display"

    ...)

    这个指令用于向终端输出用户定义的信息,包含了三种类型:

    SEND_ERROR,产生错误,生成过程被跳过。

    SATUS,输出前缀为—的信息。FATAL_ERROR,立即终止所有 cmake 过程.

    我们在这里使用的是 STATUS 信息输出,演示了由 PROJECT 指令定义的两个隐式变量

    HELLO_BINARY_DIR 和 HELLO_SOURCE_DIR。

    ADD_EXECUTABLE(hello ${SRC_LIST})

    定义了这个工程会生成一个文件名为 hello 的可执行文件,相关的源文件是 SRC_LIST 中

    定义的源文件列表

    指令是大小写无关的,参数和变量是大小写相关的。但,推荐你全部使用大写指令。

    上面的 MESSAGE 指令我们已经用到了这条规则:

    MESSAGE(STATUS “This is BINARY dir” ${HELLO_BINARY_DIR})

    这里需要特别解释的是作为工程名的 HELLO 和生成的可执行文件 hello 是没有任何关系的。

    hello 定义了可执行文件的文件名,你完全可以写成:

    ADD_EXECUTABLE(t1 main.c)编译后会生成一个 t1 可执行文件。

  4. 清理工程
    可以使用make clean清理makefile产生的中间的文件,但是,不能使用make distclean清除cmake产生的中间件。如果需要删除cmake的中间件,可以采用rm -rf ***来删除中间件。

  5. 外部构建
    在目录下建立一个build文件用来存储cmake产生的中间件,不过需要使用cmake …来运行。其中外部编译,PROJECT_SOURCE_DIR仍然指代工程路径,即/vrviu/cmake/t1,而PROJECT_BINARY_DIR指代编译路径,即/vrviu/cmake/t1/build。

三、动态库、静态库编译,安装

本节建立一个静态库和动态库,提供HelloFunc函数供其他程序编程使用,HelloFunc向终端输出Hello World字符串。安装头文件和共享库。

  1. 准备工作
    在/vrviu/cmake中建立t3,用于存放工程文件。
  2. 建立共享库
    指令:
    cd /cmake/t3
    mkdir lib
    在t2目录下建立CMakeLists.txt,内容如下:

    PROJECT(HELLOLIB)

    ADD_SUBDIRECTORY(lib)

    在lib目录下建立两个两个源文件hello.c和hello.h,

    hello.c的内容如下:

    #include "hello.h"

    void HelloFunc()

    {

            printf("Hello World\n");

    }

    hello.h的内容如下:

    #ifndef HELLO_H

    #define HELLO_H

    #include <stdio.h>

    void HelloFunc();

    #endif

    在lib的目录下建立CMakeLists.txt,内容如下:

    SET(LIBHELLO_SRC hello.c)

    SET(LIBRARY_OUTPUT_PATH ${PROJECT_BINARY_DIR}/lib)

    ADD_LIBRARY(hello SHARED ${LIBHELLO_SRC})

    ADD_LIBRARY(hello_static STATIC ${LIBHELLO_SRC})

    SET_TARGET_PROPERTIES(hello_static PROPERTIES OUTPUT_NAME "hello")

    SET_TARGET_PROPERTIES(hello PROPERTIES VERSION 1.2 SOVERSION 1)

    INSTALL(TARGETS hello hello_static LIBRARY DESTINATION lib ARCHIVE DESTINATION lib)

    INSTALL(FILES hello.h DESTINATION include/hello)

  3. 外部构建
    在build目录下:
    cmake ..
    make
    编译成功后,在build文件下的lib文件下可以发现存在一个libhello.so的动态链接库
    ADD_LIBRARY(libname [SHARED|STATIC|MODULE][EXCLUDE_FROM_ALL] source1 source2 ... sourceN)
    不需要在全libhello.so,只需要填写hello即可,cmake系统会自动为你生成libhello.X
    类型有三种:
    SHARED,动态库
    STATIC,静态库
    MODULE,在使用dyld的系统有效,如果不支持dyld,则被当做SHARED对待。
    EXCLUDE_FROM_ALL参数的意思是这个不会被默认构建,除非有其他的组件依赖或者手工构建。
  4. 添加静态库

    在以上的基础上再添加一个静态库,按照一般的习惯,则这个静态库的名字的后缀为.a。
    我们往lib/CMakeLists.txt中添加一条:

    SET_TARGET_PROPERTIES(hello_static PROPERTIES OUTPUT_NAME "hello")
    这样就可以同时得到libhello.so/libhello.a两个库了。

    ADD_LIBRARY(hello STATIC ${LIBHELLO_SRC})
    使用语句,hello作为target是不能重名的。所以会造成静态库的构建指令无效。

    SET_TARGET_PROPERTIES(target1 target2 ...PROPERTIES prop1 value1 prop2 value2 ...)
    这条指令可以用来设置输出的名称,对于动态库,还可以用来指定动态库的版本和API版本。

    与他对应的指令是:

    GET_TARGET_PROPERTY(VAR target property)
    举例:向lib/CMakeLists.txt中添加:

    GET_TARGET_PROPERTY(OUTPUT_VALUE hello_static OUTPUT_NAME)
    MESSAGE(STATUS "This is the hello_static OUTPUT_NAME:"${OUTPUT_VALUE})

    如果没有这个属性则会返回NOTFOUND.而使用以上的例子会出现一个问题,那就是会发现libhello.a存在,但是libhello.so会消失,因为cmake在构建一个新的target时,会尝试清理掉其他使用这个名字的库。解决方案如下:
    向lib/CMakeLists.txt中添加

    SET_TARGET_PROPERTIES(hello PROPERTIES CLEAN_DIRECT_PUTPUT 1)
    SET_TARGET_PROPERTIES(hello_static PROPERTIES CLEAN_DIRECT_OUTPUT 1)

    这个时候再进行构建,会发现build/lib目录中同时生成了libhello.so和libhello.a。

  5. 增加动态库的版本号

    SET_TARGET_PROPERTIES(hello PROPERTIES VERION 1.2 SOVERSION 1)
    VERSION指代动态库版本,SOVERSION指代API版本。

    此时,会生成三个文件,其中,libhello.so.1.2为动态库的文件名(realname),libhello.so.1为动态库的别名(soname),libhello.so为动态库的链接名(linkname)。

    在makefile中-lthello时,makefile会寻找libhello.so,然后将libhello.so.1写入到可执行文件的链接信息中

  6. 安装共享库和头文件
    以上面的例子,将libhello.a、libhello.so以及hello.h安装到系统目录,才能真正让其他人开发使用。例如将共享库安装到/usr/local/lib目录,将hello.h安装到/usr/local/include/hello目录。

    在lib/CMakeLists.txt中添加指令:

    INSTALL(TARGETS hello hello_static LIBRARY DESTINATION lib ARCHIVE DESTINATION lib)
    INSTALL(FILES hello.h DESTINATION include/hello)

    编译指令:

    cmake ..
    make 
    make install
    这样就可以将头文件和共享库安装到系统目录/usr/lib和/usr/local/include/hello中了

四、SDK集成

  1. 准备工作
    在cmake中创建t4用来存储这一节的资源,在t4目录下创建src目录
  2. 编码
    在src目录下编写源文件main.c如下:

    #include <hello.h>

    int main()

    {

            HelloFunc();

            return 0;

    }

    t4下的CMakeLists.txt如下:

    PROJECT(NEWHELLO)

    ADD_SUBDIRECTORY(src bin)

    t4下的src下的CMakeLists.txt如下:

    INCLUDE_DIRECTORIES(/usr/local/include/hello)

    ADD_EXECUTABLE(main main.c)

    #TARGET_LINK_LIBRARIES(main hello)

    TARGET_LINK_LIBRARIES(main libhello.a)

  3. 引入头文件搜索路径
    在src/CMakeLists.txt添加一个头文件搜索路径,如下:

    INCLUDE_DIRECTORIES(/usr/local/include/hello)

  4. 配置共享库目录

    echo /usr/local/lib” >> /etc/ld.so.conf

    #更新/etc/ld.so.cache文件

    ldconfig

  5. 编译执行
    在build目录下:
    cmake ..
    make
  6. 判断链接sdk
    指令:
    ldd src/main(在目录build下)
    静态库:

    动态库:

这篇关于Ubuntu2004 CMake 使用基础的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

python中的flask_sqlalchemy的使用及示例详解

《python中的flask_sqlalchemy的使用及示例详解》文章主要介绍了在使用SQLAlchemy创建模型实例时,通过元类动态创建实例的方式,并说明了如何在实例化时执行__init__方法,... 目录@orm.reconstructorSQLAlchemy的回滚关联其他模型数据库基本操作将数据添

Spring配置扩展之JavaConfig的使用小结

《Spring配置扩展之JavaConfig的使用小结》JavaConfig是Spring框架中基于纯Java代码的配置方式,用于替代传统的XML配置,通过注解(如@Bean)定义Spring容器的组... 目录JavaConfig 的概念什么是JavaConfig?为什么使用 JavaConfig?Jav

Java使用Spire.Doc for Java实现Word自动化插入图片

《Java使用Spire.DocforJava实现Word自动化插入图片》在日常工作中,Word文档是不可或缺的工具,而图片作为信息传达的重要载体,其在文档中的插入与布局显得尤为关键,下面我们就来... 目录1. Spire.Doc for Java库介绍与安装2. 使用特定的环绕方式插入图片3. 在指定位

Springboot3 ResponseEntity 完全使用案例

《Springboot3ResponseEntity完全使用案例》ResponseEntity是SpringBoot中控制HTTP响应的核心工具——它能让你精准定义响应状态码、响应头、响应体,相比... 目录Spring Boot 3 ResponseEntity 完全使用教程前置准备1. 项目基础依赖(M

Java使用Spire.Barcode for Java实现条形码生成与识别

《Java使用Spire.BarcodeforJava实现条形码生成与识别》在现代商业和技术领域,条形码无处不在,本教程将引导您深入了解如何在您的Java项目中利用Spire.Barcodefor... 目录1. Spire.Barcode for Java 简介与环境配置2. 使用 Spire.Barco

Java利用Spire.Doc for Java实现在模板的基础上创建Word文档

《Java利用Spire.DocforJava实现在模板的基础上创建Word文档》在日常开发中,我们经常需要根据特定数据动态生成Word文档,本文将深入探讨如何利用强大的Java库Spire.Do... 目录1. Spire.Doc for Java 库介绍与安装特点与优势Maven 依赖配置2. 通过替换

Android使用java实现网络连通性检查详解

《Android使用java实现网络连通性检查详解》这篇文章主要为大家详细介绍了Android使用java实现网络连通性检查的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录NetCheck.Java(可直接拷贝)使用示例(Activity/Fragment 内)权限要求

C# 预处理指令(# 指令)的具体使用

《C#预处理指令(#指令)的具体使用》本文主要介绍了C#预处理指令(#指令)的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录1、预处理指令的本质2、条件编译指令2.1 #define 和 #undef2.2 #if, #el

C#中Trace.Assert的使用小结

《C#中Trace.Assert的使用小结》Trace.Assert是.NET中的运行时断言检查工具,用于验证代码中的关键条件,下面就来详细的介绍一下Trace.Assert的使用,具有一定的参考价值... 目录1、 什么是 Trace.Assert?1.1 最简单的比喻1.2 基本语法2、⚡ 工作原理3

C# IPAddress 和 IPEndPoint 类的使用小结

《C#IPAddress和IPEndPoint类的使用小结》本文主要介绍了C#IPAddress和IPEndPoint类的使用小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定... 目录一、核心作用网络编程基础类二、IPAddress 类详解三种初始化方式1. byte 数组初始化2. l