c++编译过程初识

编译过程

预处理:主要是执行一些预处理指令,主要是#开头的代码,如#include 的头文件、#define 定义的宏常量、#ifdef #ifndef #endif等条件编译的代码,具体包括查找头文件、进行宏替换、根据条件编译等操作。

g++ -E example.cpp -o example.i

编译:进行词法分析、语法分析、语义分析、代码优化等,将.i文件转换为汇编代码文件,即.s文件。

g++ -S example.i -o example.s

汇编:这一阶段不同平台(Windows、Linux)的汇编器将汇编代码翻译为二进制的机器码,可被计算机识别并执行。windows上是.obj文件,linux上是.o文件。

g++ -c example.s -o example.o 

链接:将.o或.obj文件和所需的库文件组合生成最终的可执行文件。

静态链接:程序编译时,将所需的库文件直接嵌入到最终的可执行文件中,使程序在运行时不再依赖外部库文件,启动速度快,但占内存,且库文件更新需重新编译。

动态链接:程序运行时,将程序所需的库文件加载到内存中,供程序调用。节省磁盘空间,且只需要更新库文件,但外部依赖可能导致较多的环境问题。

一个C++程序可以同时包含动态链接和静态链接的部分。

g++ example.o -o exec   # 或者exec.exe文件

静态链接库 

windows上.lib文件,linux上.a文件。

// test.h
#include<iostream>
using namespace std;int cacl(int a, int b);// test.cpp
#include "test.h"int cacl(int a, int b){cout << "这是库文件内容" << endl;return a>b?a:b;
}// main.cpp
#include "test.h"int main(){int a = cacl(4, 5);cout << "这是main函数内容" << "  最大值是" << a << endl;
}

 

静态库生成:1、g++ -c test.cpp test.o  // 生成.o文件2、ar rsc libxxx.a *.o   // ar打包工具,rsc打包参数,生成.a文件3、将.a文件和头文件发给客户使用即可
静态库使用1、.a文件、头文件、测试程序(main.cpp)放一起2、g++ main.cpp -o main -L./ -lxxx    // main是生成的可执行程序。 -L指定库文件路径, -l指定库文件名称(掐头去尾中间的名称)

动态链接库 

windows上.dll文件, linux上.so文件。windows上用不同编译工具生成的动态库文件会有不同。动态库有执行权限,静态库没有执行权限;

在C++中加载动态链接库主要有两种方式:隐式加载(静态加载)和显式加载(动态加载)。上述main函数中的直接使用的方式就属于静态加载。动态加载需要用到LoadLibrary函数。

test.h test.cpp main.cpp内容同上,生成动态链接库是直接使用gcc命令并且添加-fPIC(-fpic)以及-shared参数,具体如下:

动态库生成g++ -c  -fpic test.cpp -o test.o  // -fpic生成对位置没有要求的.o文件g++ -shared test.o -o libcacl.so  // -shared生成动态库发布动态库和头文件:提供 xxx.h和 xxx.so
动态库使用g++ main.cpp -L ./ -l cacl -o app  //编译测试程序

当生成可执行程序后,通过./app执行,会报错:./app: error while loading shared libraries: libcacl.so: cannot open shared object file: No such file or directory,这是因为找不到共享库文件,还需进行以下设置:

方案1、修改LD_LIBRARY_PATH环境变量,将动态库地址添加到变量里
    vi ~/.bashrc,然后source生效
方案2、修改/etc/ld.so.conf文件
    sudo /etc/ld.so.conf  --添加动态库路径到此文件中
    sudo ldconfig       --更新/etc/ld.so.conf中的数据到/etc/ld.so.cache中
方案3、拷贝或创建软链接到/lib或/urs/lib目录
    库拷贝
    sudo cp libcalc.so /usr/lib
    创建软链接(推荐这种方式)
    sudo ln -s libcalc.so /usr/lib/libcalc.so
方案4、可执行文件内部的DT_RPATH段  --无法操作这种方案

通过以上几种方案后,无论我们把app移动到任何目录下,都可以./app执行啦~~
启动可执行程序前,可以通过一个命令检测程序能不能找到需要的动态库,这个命令是ldd
例如: ldd app,如果没有找到就会显示not found

makefile初识

以上编译时都需要执行g++命令,如果项目比较大的话,这种方式是比较麻烦的,这时候就需要makefile文件了,它是定义一系列的编译规则的文件,一旦定义好,只需要执行一个make命令,整个工程完全自动编译。

一个最简单的示例:

/*target: dependenciescommand1command2...
*/test:g++ test.cpp main.cpp -o test.PHONY:clean
clean:rm test

test就是目标文件,执行make命令时会执行“g++ test.cpp main.cpp -o test“;

.PHONY是伪目标,为了防止有clean的同名文件。clean就是伪目标,执行make clean时会执行“rm test”的命令。 

下面看一个优化后的makefile文件:

# 自定义变量
target = test
src = $(wildcard *.cpp)  # wildcard函数,查找所有的cpp文件$(target):$(CXX) $(src) -o $@   # CXX是预定义变量,$@是自动变量.PHONY:clean
clean:-rm test

makefile提供了预定义变量、自动变量,还支持自定义变量,附上两张图片,忘了之前在哪看到的了,如有侵权删。需要注意,自动变量只能在command中使用。

 

除了wildcard函数外,还有patsubst函数:(这两个函数比较常用)

patsubst : $(patsubst %.cpp, %.o $(src))
// 将src中的.cpp后缀文件改为.o后缀,这个只是makefile里的替换,并不会真的改磁盘里的文件。
// .o文件是汇编后的文件,是编译时的依赖,只是上面我们没有指定依赖,所以没用到

make执行时,会进行时间戳比较,目标文件和依赖的时间戳比较,若是目标文件比依赖靠后,那make执行会提示“已是最新”。

CMakeLists初识

CMakeLists.txt文件用来描述项目的结构和依赖关系,然后生成适合不同构建系统的项目文件,如Makefile、Visual Studio项目文件、Xcode项目文件等。所以,如果我们不想写makefile文件,就可通过CMakeLists文件来生成。

.
├── build
├── CMakeLists.txt
├── include
│   └── test.h
├── main.cpp
└── src
    └── test.cpp

下面我们通过编写CMakeLists文件,来编译代码

# cmake最低版本号
cmake_minimum_required(VERSION 3.0)
# 设置项目名称
project(cacl)# 添加头文件路径
include_directories(./include)
# 添加可执行目标
add_executable(main main.cpp)# 生成库文件
add_library(test SHARED src/test.cpp)
# 添加动态链接库
target_link_libraries(main test)

然后cd build目录,执行cmake ..,就会生成makefile文件,然后再执行make命令,就能得到可执行程序。以上编译过程实现了将test.cpp内容编译为so共享库,然后链接到main执行程序。

qmake初识

qmake是Qt官方提供的构建工具,专门用于构建Qt项目。它使用.pro文件来描述项目的结构和依赖关系,然后生成Makefile文件。 

小节 

以上知识点都比较浅显,目前我也是在学习c++的路上,希望与大家共勉~

参考链接

由浅入深,教你使用Makefile编译C++文件_c++ makefile-CSDN博客

c++ CMakeLists.txt详解_cmake --install build --strip-CSDN博客

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/65240.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Springboot高并发乐观锁

Spring Boot分布式锁的主要缺点包括但不限于以下几点&#xff1a; 性能开销&#xff1a;使用分布式锁通常涉及到网络通信&#xff0c;这会引入额外的延迟和性能开销。例如&#xff0c;当使用Redis或Zookeeper实现分布式锁时&#xff0c;每次获取或释放锁都需要与这些服务进行交…

揭秘 Fluss 架构组件

这是 Fluss 系列的第四篇文章了&#xff0c;我们先回顾一下前面三篇文章主要说了哪些内容。 Fluss 部署&#xff0c;带领大家部署Fluss 环境&#xff0c;体验一下 Fluss 的功能Fluss 整合数据湖的操作&#xff0c;体验Fluss 与数据湖的结合讲解了 Fluss、Kafka、Paimon 之间的…

leetcode82:删除链表中的重复元素II

原题地址&#xff1a;82. 删除排序链表中的重复元素 II - 力扣&#xff08;LeetCode&#xff09; 题目描述 给定一个已排序的链表的头 head &#xff0c; 删除原始链表中所有重复数字的节点&#xff0c;只留下不同的数字 。返回 已排序的链表 。 示例 1&#xff1a; 输入&…

【面试经典】多数元素

链接&#xff1a;169. 多数元素 - 力扣&#xff08;LeetCode&#xff09; 解题思路&#xff1a; 在本文中&#xff0c;“数组中出现次数超过一半的数字” 被称为 “众数” 。 需要注意的是&#xff0c;数学中众数的定义为 “数组中出现次数最多的数字” &#xff0c;与本文定…

AT24C02学习笔记

看手册&#xff1a; AT24Cxx xx代表能写入xxK bit(xx K)/8 byte 内部写周期很关键&#xff0c;代表每一次页写或字节写结束后时间要大于5ms&#xff08;延时5ms确保完成写周期&#xff09;&#xff0c;否则时序会出错。 页写&#xff1a;型不同号每一页可能写入不同大小的…

蓝牙BLE开发——解决iOS设备获取MAC方式

解决iOS设备获取MAC方式 uniapp 解决 iOS 获取 MAC地址&#xff0c;在Android、iOS不同端中互通&#xff0c;根据MAC 地址处理相关的业务场景&#xff1b; 文章目录 解决iOS设备获取MAC方式监听寻找到新设备的事件BLE工具效果图APP监听设备返回数据解决方式ArrayBuffer转16进制…

01 Oracle 基本操作

Oracle 基本操作 初使用步骤 1.创建表空间 2.创建用户、设置密码、指定表空间 3.给用户授权 4.切换用户登录 5.创建表 注意点&#xff1a;oracle中管理表的基本单位是用户 文章目录 了解Oracle体系结构 1.创建表空间**2.删除表空间**3.创建用户4.给用户授权5.切换用户登录6.表操…

【Linux命令】ps -a 和 ps -ef 的区别

ps -a 和 ps -ef 是 ps&#xff08;process status&#xff09;命令的不同选项&#xff0c;它们用于显示不同的进程信息。以下是这两个选项的主要区别&#xff1a; ps -a -a 选项表示显示所有拥有终端的进程&#xff0c;但不包括守护进程&#xff08;daemon processes&#x…

独一无二,万字详谈——Linux之文件管理

Linux文件部分的学习&#xff0c;有这一篇的博客足矣! 目录 一、文件的命名规则 1、可以使用哪些字符&#xff1f; 2、文件名的长度 3、Linux文件名的大小写 4、Linux文件扩展名 二、文件管理命令 1、目录的创建/删除 &#xff08;1&#xff09;、目录的创建 ① mkdir…

rust windwos 两个edit框

use winapi::shared::minwindef::LOWORD; use windows::{core::*,Win32::{Foundation::*,Graphics::Gdi::{BeginPaint, EndPaint, PAINTSTRUCT},System::LibraryLoader::GetModuleHandleA,UI::WindowsAndMessaging::*,}, };// 两个全局静态变量&#xff0c;用于保存 Edit 控件的…

解锁成长密码:探寻刻意练习之道

刻意练习&#xff0c;真有那么神&#xff1f; 在生活中&#xff0c;你是否有过这样的困惑&#xff1a;每天苦练英语口语&#xff0c;可一到交流时还是支支吾吾&#xff1b;埋头苦学吉他&#xff0c;却总是卡在几个和弦转换上&#xff1b;工作多年&#xff0c;业务能力却似乎陷入…

WPS中如何为指定区域的表格添加行或者列,同时不影响其它表格?

大家好&#xff0c;我是小鱼。 日常工作中会遇到这种情况&#xff1a;在一个Excel工作表中有多个表格&#xff0c;因为后期数据量增加就需要为指定区域的表格添加行或者列&#xff0c;但是不能影响其它表格。这种情况下我们应该怎么操作呢&#xff1f; 为指定区域的表格添加行…

Gitlab17.7+Jenkins2.4.91实现Fastapi项目持续发布版本详细操作(亲测可用)

一、gitlab设置&#xff1a; 1、进入gitlab选择主页在左侧菜单的下面点击管理员按钮。 2、选择左侧菜单的设置&#xff0c;选择网络&#xff0c;在右侧选择出站请求后选择允许来自webhooks和集成对本地网络的请求 3、webhook设置 进入你自己的项目选择左侧菜单的设置&#xff…

模型工作流:自动化的模型内部三角面剔除

1. 关于自动减面 1.1 自动减面的重要性及现状 三维模型是游戏、三维家居设计、数字孪生、VR/AR等几乎所有三维软件的核心资产&#xff0c;模型的质量和性能从根本上决定了三维软件的画面效果和渲染性能。其中&#xff0c;模型减面工作是同时关乎质量和性能这两个要素的重要工…

Unity微信小游戏接入开放数据域

demo地址&#xff1a;https://github.com/wechat-miniprogram/minigame-unity-webgl-transform/tree/main/Demo/Ranking 官方说明&#xff1a; https://github.com/wechat-miniprogram/minigame-unity-webgl-transform/blob/main/Design/OpenData.md 准备一个Canvas&#xff0c…

如何实现 MySQL 的读写分离?

面试题 你们有没有做 MySQL 读写分离&#xff1f;如何实现 MySQL 的读写分离&#xff1f;MySQL 主从复制原理的是啥&#xff1f;如何解决 MySQL 主从同步的延时问题&#xff1f; 面试官心理分析 高并发这个阶段&#xff0c;肯定是需要做读写分离的&#xff0c;啥意思&#x…

go window安装protoc protoc生成protobuf文件

1. 下载&#xff1a; Releases protocolbuffers/protobuf GitHub 2. 解压缩&#xff1a; 3. 配置环境变量&#xff1a; 选择系统变量->Path -> 新增 解压缩后的bin路径 4. 打印版本&#xff1a; protoc --version 5. 安装protoc-gen-go cmd 下输入安装命令&#xff0…

C#-调用C++接口

一.静态&动态装载DLL C中接口通过编译为DLL对外提供调用,C#需要将DLL加载至本应用才可实现C接口调用. 1.静态装载 C#应用程序在编译为可执行exe时将外部DLL装载至本应用中,例如在CSC编译指令中添加相关参数可实现DLL引用. csc /reference:user32.dll /out:HelloWorld.exe 2…

学习C++:标识符命名规则

标识符命名规则&#xff1a; 作用&#xff1a;C规定给标识符&#xff08;变量、常量&#xff09;命名时&#xff0c;有一套自己的规则 标识符不能是关键字 标识符只能由字母、数字、下划线组成 第一个字符必须为字母或下划线 标识符中字母区分大小写 &#xff08;给标识符命…

Linux系统程序设计--6.线程

线程基本概念 刚创建的进程默认有一个线程&#xff0c;成为主控线程(主线程) Linux线程实现 线程标识 Linux线程的创建和终止 pthread_create 龟兔赛跑案例 #include<stdio.h> #include<stdlib.h> #include<pthread.h> #include<math.h>void * th_fn…