gcc,make,cmake傻傻分不清楚?

CMake是一种跨平台编译工具,比make更为高级,使用起来要方便得多。CMake主要是编写CMakeLists.txt文件,然后用cmake命令将CMakeLists.txt文件转化为make所需要的makefile文件,最后用make命令编译源码生成可执行程序或共享库(so(shared object)).它的作用和qt的qmake是相似的。

作者:辉常哥

链接:https://www.zhihu.com/question/36609459/answer/89743845
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

1.gcc是GNU Compiler Collection(就是GNU编译器套件),也可以简单认为是编译器,它可以编译很多种编程语言(括C、C++、Objective-C、Fortran、Java等等)。

2.当你的程序只有一个源文件时,直接就可以用gcc命令编译它。

3.但是当你的程序包含很多个源文件时,用gcc命令逐个去编译时,你就很容易混乱而且工作量大

4.所以出现了make工具
make工具可以看成是一个智能的批处理工具,它本身并没有编译和链接的功能,而是用类似于批处理的方式—通过调用makefile文件中用户指定的命令来进行编译和链接的。

5.makefile是什么?简单的说就像一首歌的乐谱,make工具就像指挥家,指挥家根据乐谱指挥整个乐团怎么样演奏,make工具就根据makefile中的命令进行编译和链接的。

6.makefile命令中就包含了调用gcc(也可以是别的编译器)去编译某个源文件的命令。

7.makefile在一些简单的工程完全可以人工手下,但是当工程非常大的时候,手写makefile也是非常麻烦的,如果换了个平台makefile又要重新修改。

8.这时候就出现了Cmake这个工具,cmake就可以更加简单的生成makefile文件给上面那个make用。当然cmake还有其他功能,就是可以跨平台生成对应平台能用的makefile,你不用再自己去修改了。

9.可是cmake根据什么生成makefile呢?它又要根据一个叫CMakeLists.txt文件(学名:组态档)去生成makefile。

10.到最后CMakeLists.txt文件谁写啊?亲,是你自己手写的。

11.当然如果你用IDE,类似VS这些一般它都能帮你弄好了,你只需要按一下那个三角形
12.cmake是make maker,生成各种可以直接控制编译过程的控制器的配置文件,比如makefile、各种IDE的配置文件。
13.make是一个简单的通过文件时间戳控制自动过程、处理依赖关系的软件,这个自动过程可以是编译一个项目。

---

作者:任卫
链接:https://www.zhihu.com/question/36609459/answer/68231785
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

gcc是编译器,其实后还有连接器等
make是另外一个程序,根据makefile的指示,尽可能协调编一个程序套件的编译全过程,一个程序套件可能包好多多源代码,也可能生成若干二进制文件,还可能指示程序套件的安装程序制作和/或安装过程,那make就通过调用编译器连接器还有其他外部工具达成这一系列工作。make不光可以调用gcc编译器,也可以调用javac啊。makefile精力集中于定义工作流水线。

CMake 是另外一个更高层一点的工具,它的输出是makefile等。支持使用更简单的语言描述编译安装过程。支持多个makefile系列、VisualStudio等

下午补充下推荐使用场景:
1 gcc编译器套件:最简单啦,单文件工程比如helloworld直接使用就行了
2 make&makefile : 小中大型工程肯定要用的啦,大型工程的makefile系统不少也都是噩梦般的存在。适用于工具链【较为统一】的工程中,有一定跨平台能力。
3 CMake:若需要更为广泛的支持多编译工具链更广地跨平台且还不想构建复杂的跨平台makefile系统,那CMake确实是个很好的选择,由CMake来先编译出不同平台的makefile工程,而CMakeLists.txt集中声明描述下工程组件的关系即可。
---

从源文件

hello.c

生成(构建)可执行文件

hello

需要执行

gcc hello.c -ohello

将上面的命令写到一个文件中,这个文件就是Makefile文件,执行make会执行Makefile中的所有命令

cmake是用来生成Makefile文件的工具,生成Makefile文件的工具不止有cmake,还有autotools。Qt环境下还有qmake



作者:知乎用户
链接:https://www.zhihu.com/question/36609459/answer/190939645
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。


-----------------

Cmake知识----编写CMakeLists.txt文件编译C/C++程序

1.CMake编译原理

CMake是一种跨平台编译工具,比make更为高级,使用起来要方便得多。CMake主要是编写CMakeLists.txt文件,然后用cmake命令将CMakeLists.txt文件转化为make所需要的makefile文件,最后用make命令编译源码生成可执行程序或共享库(so(shared object))。因此CMake的编译基本就两个步骤:

1. cmake

2. make

cmake  指向CMakeLists.txt所在的目录,例如cmake .. 表示CMakeLists.txt在当前目录的上一级目录。cmake后会生成很多编译的中间文件以及makefile文件,所以一般建议新建一个新的目录,专门用来编译,例如

mkdir
 build
cd build
cmake ..

make

make根据生成makefile文件,编译程序。

2.使用Cmake编译程序

我们编写一个关于开平方的C/C++程序项目,即b= sqrt(a),以此理解整个CMake编译的过程。

a.准备程序文件

文件目录结构如下:

.
├── build
├── CMakeLists.txt
├── include
│   └── b.h
└── src
    ├── b.c
    └── main.c

头文件b.h,如下所示:

#ifndef B_FILE_HEADER_INC

#define
 B_FIEL_HEADER_INC

#include
<math.h>

double
 cal_sqrt(double
 value);


#endif

 

头文件b.c,如下所示:

#include "
../include/b.h
"


double
 cal_sqrt(double
 value)
{
    return sqrt(value);
}

 

main.c主函数,如下所示:

#include "
../include/b.h
"

#include 
<stdio.h>
int
 main(int
 argc, char
** argv)
{
    
double
 a = 49.0
; 
    
double
 b = 0.0
;

    printf(
"
input a:%f\n
"
,a);
    b 
= cal_sqrt(a);
    printf(
"
sqrt result:%f\n
"
,b);
    
return
 0
;
}

 

b.编写CMakeLists.txt

接下来编写CMakeLists.txt文件,该文件放在和src,include的同级目录,实际方哪里都可以,只要里面编写的路径能够正确指向就好了。CMakeLists.txt文件,如下所示:

 1
 #1
.cmake verson,指定cmake版本 

 2
 cmake_minimum_required(VERSION 3.2
)

 3
 
 4
 #2
.project name,指定项目的名称,一般和项目的文件夹名称对应

 5
 PROJECT(test_sqrt)

 6
 
 7
 #3
.head
 file
 path,头文件目录

 8
 INCLUDE_DIRECTORIES(

 9
 include

10
 )

11
 
12
 #4
.source directory,源文件目录

13
 AUX_SOURCE_DIRECTORY(src DIR_SRCS)

14
 
15
 #5
.set environment variable,设置环境变量,编译用到的源文件全部都要放到这里,否则编译能够通过,但是执行的时候会出现各种问题,比如"
symbol lookup error xxxxx , undefined symbol
"

16
 SET(TEST_MATH

17
 ${DIR_SRCS}

18
 )

19
 
20
 #6
.add executable file
,添加要编译的可执行文件

21
 ADD_EXECUTABLE(${PROJECT_NAME} ${TEST_MATH})

22
 
23
 #7
.add link library,添加可执行文件所需要的库,比如我们用到了libm.so(命名规则:lib+name+.so),就添加该库的名称

24
 TARGET_LINK_LIBRARIES(${PROJECT_NAME} m)

 CMakeLists.txt主要包含以上的7个步骤,具体的意义,请阅读相应的注释。

c.编译和运行程序

准备好了以上的所有材料,接下来,就可以编译了,由于编译中出现许多中间的文件,因此最好新建一个独立的目录build,在该目录下进行编译,编译步骤如下所示:

mkdir
 build
cd build
cmake ..

make

操作后,在build下生成的目录结构如下:

├── build
│   ├── CMakeCache.txt
│   ├── CMakeFiles
│   │   ├── 
3.2
.2

│   │   │   ├── CMakeCCompiler.cmake
│   │   │   ├── CMakeCXXCompiler.cmake
│   │   │   ├── CMakeDetermineCompilerABI_C.bin
│   │   │   ├── CMakeDetermineCompilerABI_CXX.bin
│   │   │   ├── CMakeSystem.cmake
│   │   │   ├── CompilerIdC
│   │   │   │   ├── a.out
│   │   │   │   └── CMakeCCompilerId.c
│   │   │   └── CompilerIdCXX
│   │   │       ├── a.out
│   │   │       └── CMakeCXXCompilerId.
cpp

│   │   ├── cmake.check_cache
│   │   ├── CMakeDirectoryInformation.cmake
│   │   ├── CMakeOutput.log
│   │   ├── CMakeTmp
│   │   ├── feature_tests.bin
│   │   ├── feature_tests.c
│   │   ├── feature_tests.cxx
│   │   ├── Makefile2
│   │   ├── Makefile.cmake
│   │   ├── progress.marks
│   │   ├── TargetDirectories.txt
│   │   └── test_sqrt.
dir

│   │       ├── build.
make

│   │       ├── C.includecache
│   │       ├── cmake_clean.cmake
│   │       ├── DependInfo.cmake
│   │       ├── depend.internal
│   │       ├── depend.
make

│   │       ├── flags.
make

│   │       ├── link.txt
│   │       ├── progress.
make

│   │       └── src
│   │           ├── b.c.o
│   │           └── main.c.o
│   ├── cmake_install.cmake
│   ├── Makefile
│   └── test_sqrt
├── CMakeLists.txt
├── include
│   └── b.h
└── src
    ├── b.c
    └── main.c

注意在build的目录下生成了一个可执行的文件test_sqrt,运行获取结果如下:

命令:
.
/test_sqrt 
结果:
input a:
49.000000

sqrt result:
7.000000

 

d.源码

地址test_sqrt.tar.gz

 

3.参考资料

[1]. CMake 使用方法 & CMakeList.txt


<think>嗯,用户想将现有的Make项目迁移到CMake,我需要提供最佳实践和步骤。首先,我应该回忆一下CMake的基本结构和常用命令。CMake使用CMakeLists.txt文件来定义构建过程,而Make用的是Makefile。用户可能对CMake不熟悉,所以需要一步步引导。 首先,可能需要比较MakeCMake的不同。比如,Make的变量通常是直接赋值的,而CMake变量有作用域,区分大小写,这点引用[1]提到过。然后,用户需要创建一个顶层的CMakeLists.txt,指定项目名称、版本和语言,比如C或C++。然后,设置变量,比如SOURCE_FILES,包含所有的源文件,这可能和Makefile中的SRCS变量类似。 接下来,添加可执行文件或库。用户原来的Makefile可能用了gcc直接编译,现在换成add_executable。然后,处理依赖关系。Make中使用目标依赖,而CMake可以用target_link_libraries来链接库,比如数学库需要链接m,这可能要用到target_link_libraries(... m)。 然后是安装和测试。Make中可能有install目标,CMake中使用install命令,指定安装路径。测试的话,引用[2]提到Ctest,所以需要enable_testing()和add_test,这部分需要详细说明。 迁移步骤的话,应该建议逐步迁移,比如先处理主程序,再处理子目录。可能还需要处理条件编译,比如在Make中使用ifeq,CMake中用option或if命令。比如用户之前用DEBUG=1,在CMake中用option(DEBUG "Debug mode" ON)然后根据变量设置编译选项。 另外,生成器表达式可能对条件编译有帮助,比如根据配置设置不同的编译器标志。还有外部依赖,比如find_package,用户原来的Make可能直接指定路径,现在可以用find_library或者find_package来处理。 常见问题部分,用户可能会遇到变量作用域的问题,比如父作用域的变量在子目录不可见,需要用PARENT_SCOPE。或者如何传递编译器选项,用target_compile_options。还有如何处理多个构建目录,用out-of-source构建,比如建立build目录。 教程推荐方面,官方教程和现代CMake实践可能有帮助。最后,用户可能需要转换工具,比如cmake-converter,但提醒他们可能需要手动调整。 过程中要确保LaTeX格式正确,比如变量名用$VARIABLE$,独立公式用$$...$$。同时生成相关问题,比如迁移中的依赖处理、性能对比、调试方法,以及结合CTest和CDash的用法。</think>### 从Make迁移到CMake的步骤与最佳实践 #### 一、核心差异与迁移思路 1. **语法与变量管理** Make使用基于规则的语法和全局变量(如`CC=gcc`),而CMake通过`CMakeLists.txt`文件声明式管理变量,且变量作用域受控(如`set(VARIABLE value)`)[^1]。 $$ \text{示例:} \quad \text{Make中的} \quad CC = gcc \quad \rightarrow \quad \text{CMake中的} \quad \text{set(CMAKE_C_COMPILER "gcc")} $$ 2. **构建目标定义** Make通过目标(如`all: main.o`)和隐式规则定义构建流程,CMake需显式声明目标(如`add_executable(main src/main.c)`)并管理依赖关系。 #### 二、分步迁移指南 1. **创建顶层CMakeLists.txt** ```cmake cmake_minimum_required(VERSION 3.10) project(MyProject VERSION 1.0 LANGUAGES C CXX) set(CMAKE_CXX_STANDARD 11) ``` 2. **定义源文件与目标** - 替换Make中的`SRCS = main.c utils.c`: ```cmake file(GLOB SOURCE_FILES src/*.c) add_executable(myapp ${SOURCE_FILES}) ``` 3. **处理依赖关系** - Make中的`-lm`链接数学库: ```cmake target_link_libraries(myapp PRIVATE m) ``` 4. **条件编译与变量传递** - 替换Make的`DEBUG=1`: ```cmake option(DEBUG "Enable debug mode" OFF) if(DEBUG) target_compile_definitions(myapp PRIVATE DEBUG_MODE=1) endif() ``` 5. **多目录项目处理** - 使用`add_subdirectory()`拆分模块: ```cmake add_subdirectory(libutils) target_link_libraries(myapp PRIVATE utils) ``` #### 三、高级配置 1. **安装规则迁移** Make中的`install`目标转换为: ```cmake install(TARGETS myapp DESTINATION bin) install(DIRECTORY include/ DESTINATION include) ``` 2. **集成测试(CTest)** 结合引用[2]的测试方法: ```cmake enable_testing() add_test(NAME basic_test COMMAND myapp --test) ``` #### 四、验证与调试 1. **生成构建系统** ```bash mkdir build && cd build cmake .. -DDEBUG=ON cmake --build . ``` 2. **常见问题解决** - **变量未传递**:使用`PARENT_SCOPE`修饰符 - **链接失败**:通过`find_library(MATH_LIB m)`显式查找库 #### 五、推荐教程与工具 1. **官方资源** - [CMake官方教程](https://cmake.org/cmake/help/latest/guide/tutorial/index.html) - 《Modern CMake for C++》 2. **自动化工具** 工具如`cmake-converter`可辅助转换,但需手动调整复杂逻辑。
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值