首页 > 系统相关 >linux开发C/C++

linux开发C/C++

时间:2023-09-02 16:11:26浏览次数:36  
标签:++ C++ 编译 add 开发 linux cpp main

最近在部署项目的时候总是会遇到关于C++的编译问题,由于之前学习C++只是为了参加算法竞赛,缺少这一部分的知识,所以学习一下这一相关内容,并做一下记录

参考:VSCode开发C++七讲【基于VSCode和CMake实现C/C++开发 | Linux篇】https://www.bilibili.com/video/BV1fy4y1b7TC?p=17&vd_source=7f1982c56a437c7b00c3695687b7086d

linux上安装C/C++的开发环境

  1. 安装编译器gcc,调试器gdb
  • 安装命令:
sudo apt-get install build-essential gdb
  • 检验安装:(通过查看版本号,如果正常显示则说明安装成功)
gcc --version
g++ --version
gdb --version	
  1. 安装cmake
  • 安装命令:
sudo apt-get install cmake
  • 检验安装
cmake --version

GCC编译

GCC编译器是一种功能强大且自由的编程语言编译器,适用于Linux系统,能高效地编译和构建各种类型的软件。

一般使用GCC编译C代码,使用G++编译C++代码。

下面以G++编译C++代码为例子:

编译过程

在linux上使用G++编译C++的时候,输入的代码为: g++ a.cpp -o a,即可生成a.cpp对应的可执行文件,但是实际上这个代码包含着下面四个步骤。

  1. 预处理
g++ -E a.cpp -o a.i  # 生成.i文件
  1. 编译
g++ -S a.i -o a.s # 生成.s文件
  1. 汇编
g++ -C a.s -o a.o  # 生成.o文件
  1. 链接
g++ a.o -o a  # 生成可执行的bin文件

重要参数

参考:http://t.csdn.cn/atKWQ

列举几个常用的:

  1. -g 编译带调试信息的可执行文件
  2. -O 优化代码,加快执行等等 默认为O1 除此以外还有O0(不优化) / O1 / O2(最多用O2) / O3
  3. -l(小写的L) 指定库文件,后面紧跟着库名 在/usr/lib 、/lib、/usr/local/lib里面的库直接用-l参数链接
g++ -lglog a.cpp
  1. -L 指定库文件路径 如果想链接的库不在上面的三个目录中,需要指定路径
# 例如:想链接/home/ubuntu/下my库文件
g++ -L/home/ubuntu/ -lmy a.cpp
  1. -I(大写I) 指定头文件搜索目录 如果头文件在/usr/include 目录下就不需要指定,默认会去那里找

    如果头文件不在,需要利用-I进行指定,指定的时候可以使用相对路径

# 例如:在/home/ubuntu下面有a.cpp中导入的头文件
g++ -I/home/ubuntu a.cpp
  1. -Wall 打印警告信息

  2. -W 关闭警告信息

  3. -std=c++11 设置C++11的编译标准

  4. -o 指定输出的文件名

  5. -D 定义宏

编译生成库文件

# 最初目录结构
├── include
│   └── Swap.h
├── main.cpp
└── src
    └── Swap.cpp
# 文件关系
main.cpp调用Swap.cpp
Swap.cpp引用了Swap.h的头文件

编译生成静态库并进行链接生成可执行文件:

# 汇编生成Swap.o文件
g++ Swap.cpp -c -I../include
# 生成静态库libSwap.a
ar rs libSwap.a Swap.o
# 链接静态库并生成可执行文件:staticmain
g++ main.cpp -Iinclude -Lsrc -lSwap -o staticmain

编译生成动态库并进行链接生成可执行文件:

# 生成动态库libSwap.so
g++ Swap.cpp -I../include -fPIC -shared -o libSwap.so
## 上面命令等价于以下两条命令
# gcc Swap.cpp -I../include -c -fPIC
# gcc -shared -o libSwap.so Swap.o
# 链接生成可执行文件:sharemain
g++ main.cpp -Iinclude -Lsrc -lSwap -o sharemain

CMake

  • CMake是一个跨平台的安装编译工具,可以用简单的语句来描述所有平台的安装(编译过程)。
  • CMake可以说已经成为大部分C++开源项目标配

应用场景:对于一个工程,如果我们在工程中额外的添加了一个C++代码,对于不同的平台我们要进行不同的操作,而如果使用了CMake进行构建这个工程,额外的添加一个C++代码,我们只需要对CMakeLists.txt进行修改,在其他不同的平台上不需要进行不同的额外操作。

CMake语法介绍

  • 基本语法格式:指令(参数 1 参数 2…)

    • 参数使用括弧括起
    • 参数之间使用空格或分号分开
  • 指令是大小写无关的,参数和变量是大小写相关的

# 指令大小写无关 set与SET作用一样
set(HELLO hello.cpp) #  设置变量HELLO值是hello.cpp
SET(HELLO hello.cpp)
# 参数和变量是大小写相关的
add_executable(hello main.cpp HELLO.cpp) # 上面的变量HELLO是大写,这里想引用也一定要大写
  • 变量使用${}方式取值,但是在if控制语句中是直接使用变量名
# 变量使用${}方式取值
ADD_EXECUTABLE(hello main.cpp ${HELLO}) 

# IF 控制语句中是直接使用变量名
if(HELLO)
if(${HELLO}) # 用法错误

重要指令

  • cmake_minimum_required - 指定CMake的最小版本要求 (一般放在第一行进行指定)

    语法:cmake_minimum_required(VERSION 指定的最小版本号 [如果不符合的报错信息])

# 例如:CMake最小版本要求为2.8.3
cmake_minimum_required(VERSION 2.8.3)
  • project - 定义工程名称,并可指定工程支持的语言

    语法:project(工程名称 [工程支持的语言1】 [工程支持的语言2])

# 例如:指定工程名为HELLOWORLD
project(HELLOWORLD)
  • set - 显式的定义变量

    语法:set(变量 [变量值1】[变量值2])

# 例如:定义SRC变量,其值为main.cpp hello.cpp
set(SRC sayhello.cpp hello.cpp)
  • include_directories - 向工程添加多个特定的头文件搜索路径 --->相当于指定g++编译器的-I参数

    语法:include_directories(路径1 路径2 …)

# 例如:将/usr/include/myincludefolder 和 ./include 添加到头文件搜索路径
include_directories(/usr/include/myincludefolder ./include)
  • link_directories - 向工程添加多个特定的库文件搜索路径 --->相当于指定g++编译器的-L参数

    语法:link_directories(路径1 路径2 …)

# 例如:将/usr/lib/mylibfolder 和 ./lib 添加到库文件搜索路径
link_directories(/usr/lib/mylibfolder ./lib)
  • add_library - 生成库文件

    语法:add_library(libname [指定库的类型:SHARED动态库|STATIC静态库|][ 资源1 资源2 … )

# 例如:通过变量 SRC其值为main.cpp hello.cpp 生成 libhello.so(动态库) 共享库
add_library(hello SHARED ${SRC})
  • add_compile_options - 添加编译参数

    语法:add_compile_options(参数1 参数2 参数3 ....)

# 例如:添加编译参数 -Wall -std=c++11 -O2
add_compile_options(-Wall -std=c++11 -O2)
  • add_executable - 生成可执行文件

    语法:add_library(可执行文件名称 待编译的代码1 待编译的代码2 … )

# 例如:编译main.cpp生成可执行文件main
add_executable(main main.cpp)
  • target_link_libraries - 为 target 添加需要链接的共享库 --->相同于指定g++编译器-l参数

    语法:target_link_libraries(目标文件 链接的库1 链接的库2…)

# 例如:将前面add_library生成的hello动态库文件链接到add_executable生成的可执行文件main
target_link_libraries(main hello)
  • add_subdirectory - 向当前工程添加存放源文件的子目录,并可以指定中间二进制和目标二进制存放的位置

    语法:add_subdirectory(要添加的源文件目录)

# 例如:添加src子目录,一定要确保src中需有一个CMakeLists.txt
add_subdirectory(src)
  • aux_source_directory - 发现一个目录下所有的源代码文件并将列表存储在一个变量中,这个指令临时被用来自动构建源文件列表

    语法:aux_source_directory(文件所在的目录的路径 变量名)

#例如:定义SRC变量,其值为当前目录(.)下所有的源代码文件
aux_source_directory(. SRC) 
# 编译SRC变量所代表的源代码文件,生成main可执行文件
add_executable(main ${SRC})

CMake常用变量

  • CMAKE_C_FLAGS gcc编译选项CMAKE_CXX_FLAGS g++编译选项
# 例如:在CMAKE_CXX_FLAGS编译选项后追加-std=c++11
set( CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11")
  • CMAKE_BUILD_TYPE 编译类型(Debug, Release)
# 常用到
# 设定编译类型为debug,调试时需要选择debug
set(CMAKE_BUILD_TYPE Debug) 
# 设定编译类型为release,发布时需要选择release
set(CMAKE_BUILD_TYPE Release) 
  • CMAKE_C_COMPILER:指定C编译器
  • CMAKE_CXX_COMPILER:指定C++编译器
  • EXECUTABLE_OUTPUT_PATH:可执行文件输出的存放路径
  • LIBRARY_OUTPUT_PATH:库文件输出的存放路径

CMake编译工程

CMake目录结构:项目主目录存在一个CMakeLists.txt文件

两种方式设置编译规则

  1. 包含源文件的子文件夹包含CMakeLists.txt文件,主目录的CMakeLists.txt通过add_subdirectory添加子目录即可;
  2. 包含源文件的子文件夹未包含CMakeLists.txt文件,子目录编译规则体现在主目录的CMakeLists.txt中;

编译流程

在 linux 平台下使用 CMake 构建C/C++工程的流程如下:

  • 手动编写 CmakeLists.txt。
  • 执行命令 cmake PATH生成 Makefile ( PATH 是顶层CMakeLists.txt 所在的目录 )。
  • 执行命令make 进行编译。

两种构建方式

  • 内部构建

    内部构建会在同级目录下产生一大堆中间文件,这些中间文件并不是我们最终所需要的,和工程源文件放在一起会显得杂乱无章。

## 内部构建
# 在当前目录下,编译本目录的CMakeLists.txt,生成Makefile和其他文件
cmake .
# 执行make命令,生成target
make
  • 外部构建:经常遇到 超级重要!!!!!

    将编译输出文件与源文件放到不同目录中

## 外部构建
# 1. 在当前目录下,创建build文件夹
mkdir build 
# 2. 进入到build文件夹
cd build
# 3. 编译上级目录的CMakeLists.txt,生成Makefile和其他文件
cmake ..
# 4. 执行make命令,生成target
make

CMake案例

反复看【基于VSCode和CMake实现C/C++开发 | Linux篇】https://www.bilibili.com/video/BV1fy4y1b7TC?p=19&vd_source=7f1982c56a437c7b00c3695687b7086d !!!

标签:++,C++,编译,add,开发,linux,cpp,main
From: https://www.cnblogs.com/ambitionx/p/17673798.html

相关文章

  • Linux语法注意区分千万别记混!!!注意!!!
    Docker相关语法删除容器:dockerrm-f  容器名或容器ID删除镜像:dockerrmi-f  镜像名或镜像id导入镜像文件到本地dockerload-i  镜像压缩包文件名清理已关闭容器: docker  container prune查看已关闭容器:docker  ps  -a  查看数据卷: docker  volum......
  • 深入理解linux系统目录
    / 根目录。 包含了几乎所的文件目录。相当于中央系统。进入的最简单方法是:cd/。/boot 引导程序,内核等存放的目录。 这个目录,包括了在引导过程中所必需的文件,引导程序的相关文件(例如grub,lilo以及相应的配置文件以及Linux操作系统内核相关文件(例如vmlinuz等一般都存放在这里。在......
  • 【WCH蓝牙系列芯片】-基于CH582开发板—基础外设输出PWM波形讲解
    ---------------------------------------------------------------------------------------------------------------------------------------------------------------------在WCH官方提供的CH583的EVT资源包中,我们可以找到PWMX的例程,这是一个8位的PWM输出,占空比和周期可调的......
  • 基础linux命令
    前言:由于在实际开发过程中服务器大多部署在linux系统下,所以特此来学习linux的基本操作1.1pwdpwd命令的目的是打印当前目录,告诉你目前在哪里比如我在kali终端中输入pwd,实际返回为/home/kali1.2lsls可以列出当前目录下有什么文件当然也可以查看不同目录的内容,比如ls文件地......
  • 虚拟机VMware与乌班图的安装 -- 正点原子嵌入式Linux学习
    一、准备工作1、虚拟机VMware的下载官网下载地址:DownloadVMwareWorkstationPro2、linux乌班图的下载官网下载地址:下载Ubuntu桌面系统|Ubuntu二、虚拟机VMware的安装过程1、点击第一步下载好的虚拟机安装文件,选择自定义,后点击下一步2、点击稍后安装3、选择Linux......
  • linux自带i2c工具使用
    I2C总线被全球超过50个公司的1000+个ICs所使用,已然是一个世界标准.另外,I2C总线与多种不同的控制总线是兼容的,比如SMBus(系统管理总线),PMBus(电源管理总线),IPMI(智能平台管理总线),DDC(显示数据通道)以及ATCA(高级电信架构).如果没记错的话,linux中的I2C框架是完全支持SMBu......
  • 如何扩容Linux文件系统?看这里↓↓↓
    一、在虚拟机中添加一块新磁盘注意:如果是未关机的状态下,添加完新的虚拟磁盘需要重启才能生效查看当前系统内的磁盘数量,确认我们新添加的磁盘能正常显示了fdisk-l二、为新添加的磁盘创建分区fdisk/dev/sdb命令(输入m获取帮助):n #输入n创建分区分区类型p主分区(0......
  • Linux中进程相关的API
    在Linux中,进程控制相关的API非常多。以下是一些常用的进程控制相关的系统调用(syscalls)和库函数:创建和终止进程:fork():创建一个新进程,这是创建新进程的最常用方法。vfork():类似于fork(),但有一些差异,主要用于exec调用之前。exec():系列函数(如execl(),execp(),execle()......
  • python flask 提供web的get/post开发
    转载请注明出处:使用pythonflask框架编写webapi中的get与post接口,代码编写与调试示例如下:fromflaskimportFlask,request,jsonifyapp=Flask(__name__)@app.route('/api/get',methods=['GET'])defhandle_get_request():try:#解析URL参数......
  • Linux安装fail2ban
    1、环境:centos7 2、安装fail2banyum-yinstallepel-release#epel库里就有fail2ban直接安装epel库就可以使用yum-yinstallfail2ban#安装fail2ban3、配置fail2ban贴上官方文档:http://www.fail2ban.org/wiki/index.php/MANUAL_0_......