了解C++工作机制

news2024/9/25 23:12:38

基于hello.cpp对C++的运行进行一个初步认识,并介绍国外C++大佬Cherno常用的项目结构和调试Tips

C++是如何工作的

  • C++工作流程
  • 1.实用工程(project)结构
    • (1)Microsoft Visual Studio2022新建项目后,自动生成的原始文件和文件夹:
    • (2)写一个简单的main.cpp,编译运行后,自动创建文件加和生成中间文件(obj)和可执行文件(exe)
    • (3)从默认工程/项目(project)结构到实用工程/项目结构
  • 2.调试(debug)
    • 3.1整体代码
    • 3.2预处理
    • 3.3.编译
    • 3.4链接
  • 4.C++编译器
    • 4.1定义与作用
    • 4.2示例
    • 4.3编译器的优化代码例子 ——“常数折叠”
  • 5.C++链接器
  • 6.头文件
    • 6.1定义与作用
    • 6.2示例
    • 4.3在实现函数的 functions.cpp 文件中包含 #include "functions.h" 的目的

C++工作流程

1.实用工程(project)结构

(1)Microsoft Visual Studio2022新建项目后,自动生成的原始文件和文件夹:

在这里插入图片描述

(2)写一个简单的main.cpp,编译运行后,自动创建文件加和生成中间文件(obj)和可执行文件(exe)

在这里插入图片描述

在这里插入图片描述

(3)从默认工程/项目(project)结构到实用工程/项目结构

  1. 实用如下修改输出目录和中间目录
$(SolutionDir)bin\$(Platform)\$(Configuration)\
$(SolutionDir)bin\intermediates\$(Platform)\$(Configuration)\
-$(SolutionDir)代表当前解决方案,不同的的解决方案会进行不同的具体的实例化,$(Platform)以此类推

在这里插入图片描述
在这里插入图片描述
2. 切换为“显示所有文件”的视图,并且新建一个src文件夹,用于收录.cpp文件
在这里插入图片描述
在这里插入图片描述

3.写一个简单的cpp文件,并且查看编译运行后的目录结构
在这里插入图片描述
在这里插入图片描述

这样的目录结构是不是更清晰喜人呢?

2.调试(debug)

在这里插入图片描述
Ctrl+F5进入调试模式,F11可单步执行,调试–>窗口–>内存–>内存1,可以查看当前程序使用的内存的内存的状态
在这里插入图片描述

断点之上的代码已经运行,断点及断点之下的代码将要运行,这里我们键入参数a的地址,并且回车
在这里插入图片描述

在这里插入图片描述

3.1整体代码

#include<iostream>

int main()
{
	std::cout << "hello, cpp!" << std::endl;
	std::cin.get();
	return 0;
}
}

3.2预处理

预处理(Preprocessing)是 C++ 编译过程中的第一个阶段,它在实际的编译之前对源代码进行一系列的文本替换和处理操作。

#include <iostream>

在编译之前,C++ 编译器会进行预处理,经预处理后,iostream文件中的内容就被copy到我们的源文件中了。
项目中每一个cpp文件都会被编译,但头文件不会被编译,因为其已经被预处理过了。

3.3.编译

编译是将高级程序设计语言(如C++)编写的源代码转换成计算机可以执行的机器代码的过程。

cpp的代码正常情况下是从上到下一行一行编译的(头文件不会被执行编译,其在预处理时被包含进来,和后面的代码一起编译)。

#include<iostream> 

int main()
{
	std::cout << "hello, cpp!" << std::endl;
	std::cin.get();
	return 0}

完成编译后,生成了obj文件。在 Windows 平台上,“.obj” 文件通常是使用 Microsoft Visual Studio 编译器生成的。

在这里插入图片描述

obj文件包含了编译器对源代码的翻译结果。 这个文件中包含了与源代码中的每个函数和变量相关的机器代码,并且还可能包含一些调试信息、符号表、重定位信息等。目标文件的生成是编译的一部分,它还需要经过链接过程,与其他目标文件和库文件一起生成最终的可执行文件(exe)。

3.4链接

链接(Linking)是 C++编译过程中的一个重要阶段,它将编译器生成的目标文件和其他可能用到的目标文件或库文件合并成最终的可执行文件。

在这里插入图片描述

每一个cpp源文件都对应一个obj,链接器将其连接后,生成可执行文件。

增加一个cpp源文件,如下:

#include<iostream>

void log(const char* message)
{
	std::cout << message << std::endl;
}

对其编译后,在debug文件夹下会看到其生成的obj文件:

在这里插入图片描述

4.C++编译器

4.1定义与作用

C++编译器是一种将C++源代码翻译成目标代码(通常是机器代码)的工具。其工作包括词法分析、语法分析、语义分析、优化和代码生成等阶段。

  • 词法分析(Lexical Analysis):编译器首先会将源代码分解成基本单元,这些单元被称为词法单元(tokens),比如关键字、标识符、运算符等。

  • 语法分析(Syntax Analysis):在这个阶段,编译器将词法单元组织成语法树,以检查代码的结构是否符合语法规则。

  • 语义分析(Semantic Analysis):编译器会检查代码的语义是否合法,比如变量是否被正确声明和使用,函数是否正确调用等。

  • 优化(Optimization):在生成目标代码之前,编译器可能会进行一系列的优化操作,以提高程序的性能或减小生成的目标代码的体积。

  • 代码生成(Code Generation):最终阶段是生成目标代码,这是计算机可以直接执行的机器代码。这些代码通常是针对特定硬件架构的。

4.2示例

#include <iostream>

int main() {
    int x = 5; // 变量声明和赋值
    int y = 10;
    int result = x + y; // 加法操作

    std::cout << "The result is: " << result << std::endl;

    return 0;
}

在这个示例中,编译器将首先进行词法分析,将代码分解成词法单元。然后进行语法分析,构建语法树。接着进行语义分析,确保变量被正确声明和使用。最后,编译器会生成目标代码,执行加法操作并输出结果。这是一个简化的过程,实际编译器会进行更多的优化和处理。

4.3编译器的优化代码例子 ——“常数折叠”

常数折叠(Constant Folding)是指编译器在编译阶段对常量表达式进行计算,并将其结果直接替换回表达式的值。这个过程发生在编译器静态分析阶段,有助于优化代码,减少运行时的计算量。

举个例子:

int result = 5 + 3 * 2;

在常数折叠过程中,编译器会对表达式进行计算。在这个例子中,3 * 2是一个常量表达式,编译器可以在编译阶段计算出其结果为6。因此,整个表达式可以被折叠为:

int result = 5 + 6;

这个过程可以减少程序运行时的计算量,提高代码的执行效率。

5.C++链接器

链接器(Linker)是编译器工具链中的一个部分,它负责将多个目标文件(Object Files)或者库文件(Library Files)中的代码和数据结合起来,生成最终的可执行文件或者库文件。在程序编译过程中,源代码会被编译成目标文件,这些目标文件可能包含了程序的不同部分,比如函数、变量等。

链接器的主要任务包括:

  • 符号解析(Symbol Resolution):处理各个目标文件中使用的符号(如函数、变量名),将其与实际的内存地址或者其他目标文件中的符号联系起来。

  • 符号合并(Symbol Combining):将各个目标文件中定义的符号整合到最终的输出文件中,解决多个目标文件中可能存在的重复定义。

  • 地址重定位(Address Binding):将各个目标文件中的符号地址绑定到最终的可执行文件或库文件中,以便在程序加载到内存并执行时,能够正确地定位和访问各个符号。

  • 生成可执行文件或库文件(Executable or Library Generation):最终将链接后的目标文件生成可执行文件(例如.exe文件)或者库文件(例如.dll或.lib文件),供操作系统或其他程序使用。

链接器的工作分为静态链接和动态链接两种方式。静态链接是将所有的目标文件和库文件在编译时链接成一个完整的可执行文件,而动态链接是在程序运行时才进行链接,可以实现共享库(Shared Library)的功能。

总的来说,链接器在编译过程的最后阶段,将各个模块或库文件整合在一起,生成可执行的程序或者库文件,使得程序能够正确地运行。

6.头文件

6.1定义与作用

头文件(Header File)是一种文本文件,通常以.h为扩展名,包含了函数、类、变量等的声明和定义。头文件的主要作用是提供接口和声明,让程序能够访问和使用某些功能而无需了解其具体实现细节。

头文件的作用包括:

  • 声明和接口提供: 头文件通常包含函数、类、变量的声明,允许在不暴露实现细节的情况下使用这些元素。其他文件可以包含头文件,从而访问其中声明的内容,而不必了解其实现细节。

  • 模块化和组织性: 头文件有助于组织代码,将相关的功能组织到单独的模块中,并通过包含不同的头文件来实现模块化。这样可以提高代码的可维护性和可读性。

  • 重用和共享代码: 头文件中的声明和接口可以被多个文件共享和重用,使得相同的函数、类或变量能够在多个文件中使用,减少重复编写代码的工作量。

头文件在以下场景中能够发挥作用:

  • 多文件项目: 当程序由多个源文件组成时,头文件能够提供接口和声明,使得各个文件能够互相访问和使用彼此的功能,提高了代码的组织性和可维护性。

  • 库开发: 在开发库或框架时,头文件对外提供了公共接口和声明,允许其他开发者使用库的功能而无需了解其内部实现。

  • 项目协作: 在团队协作开发项目时,头文件定义了函数、类等的接口,让团队成员了解如何使用这些接口而无需深入了解实现细节。

6.2示例

创建头文件(例如:functions.h):头文件包含函数的声明和可能的结构、常量或其他相关内容。

// functions.h
#pragma once

// 函数声明
int add(int a, int b);
float divide(float a, float b);

在源文件中包含头文件:在需要使用这些函数的源文件中,包含头文件。

// main.cpp

#include <iostream>
#include "functions.h" // 包含头文件

int main() {
    int result = add(5, 3); // 调用头文件中声明的函数
    std::cout << "Result: " << result << std::endl;
    return 0;
}

实现函数:在源文件中实现头文件中声明的函数。

// functions.cpp

#include "functions.h" // 包含对应的头文件

int add(int a, int b) {
    return a + b;
}

float divide(float a, float b) {
    if (b != 0) {
        return a / b;
    } else {
        return 0; // 可以根据实际需求进行异常处理或错误提示
    }
}

好处:

  • 模块化和组织性: 使用头文件可以将函数的声明和结构、常量等信息组织在一起,使得代码更模块化和易于维护。

  • 可读性: 头文件提供了函数接口的清晰定义,使得阅读和理解代码更为容易。

  • 代码重用: 通过头文件的方式,可以在多个源文件中重复使用相同的函数声明,提高了代码的重用性。

  • 错误检查: 预编译器指令(#ifndef、#define、#endif)用于防止头文件被多次包含,避免了重复定义和编译错误。

  • 维护方便: 当需要修改函数接口时,只需在头文件中修改函数的声明,而不必修改所有使用该函数的源文件。

4.3在实现函数的 functions.cpp 文件中包含 #include “functions.h” 的目的

在实现函数的 functions.cpp 文件中包含 #include “functions.h” 是为了确保函数的实现与其声明一致。这样做有以下作用:

  • 检查一致性: #include “functions.h” 会将头文件中的函数声明包含到 .cpp 文件中,这样编译器就能检查函数的实现是否与其声明一致。如果实现与声明不一致,编译器会报错,提醒修正错误。

  • 可读性和维护性: 包含头文件可以让读者清晰地看到函数的接口和声明。这样使得代码更易读、易懂,并且方便维护。

  • 代码一致性: 保持函数声明和定义的一致性是良好编程实践的一部分,有助于代码的可靠性和一致性。

在某些情况下,如果函数的实现非常简单,不需要使用头文件中的其他信息(例如结构、常量等),有时可以省略包含头文件。但是,为了代码的可读性和一致性,我们最好在函数的实现文件中包含对应的头文件。

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

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

相关文章

【机器学习】密度聚类:从底层手写实现DBSCAN

【机器学习】Building-DBSCAN-from-Scratch 概念代码数据导入实现DBSCAN使用样例及其可视化 补充资料 概念 DBSCAN&#xff08;Density-Based Spatial Clustering of Applications with Noise&#xff0c;具有噪声的基于密度的聚类方法&#xff09;是一种基于密度的空间聚类算…

脉冲水表计量方法有哪些?

随着社会的发展和人们对资源的需求不断增长&#xff0c;水资源的有效利用显得尤为重要。而水表作为测量家庭和工业用水的关键设备&#xff0c;其准确性和稳定性对于水资源管理至关重要。本文将重点介绍水表脉冲数计量方法&#xff0c;以帮助读者更好地理解和应用水资源。 一、机…

Magnific AI:彻底改变 AI 生成图像的升级

在我最近与 Magnific AI 的讨论中&#xff0c;我不仅感到惊讶&#xff0c;而且对该工具提供的质量和可能性着迷。我发现 Magnific AI 能够转换人工智能生成的图像&#xff08;这些图像通常只能以低分辨率提供&#xff09;&#xff0c;尤其令人印象深刻&#xff0c;不仅在可打印…

【MATLAB第84期】基于MATLAB的波形叠加极限学习机SW-ELM代理模型的sobol全局敏感性分析法应用

【MATLAB第84期】基于MATLAB的波形叠加极限学习机SW-ELM代理模型的sobol全局敏感性分析法应用 前言 跟往期sobol区别&#xff1a; 1.sobol计算依赖于验证集样本&#xff0c;无需定义变量上下限。 2.SW-ELM自带激活函数&#xff0c;计算具有phi&#xff08;x&#xff09;e^x激…

Kafka基本原理及使用

目录 基本概念 单机版 环境准备 基本命令使用 集群版 消息模型 成员组成 1. Topic&#xff08;主题&#xff09;&#xff1a; 2. Partition&#xff08;分区&#xff09;&#xff1a; 3. Producer&#xff08;生产者&#xff09;&#xff1a; 4. Consumer&#xff08;…

多任务学习(Multi-task Learning)的详细过程和具体示例

多任务学习 1.多任务学习的基本思想2. 多任务学习的过程包括:3. 多任务学习的具体例子: 1.多任务学习的基本思想 多任务学习(Multi-task Learning)是一种机器学习的方法,它可以同时解决多个相关任务。 多任务学习的基本思想是:多个任务之间存在一定的相关性或公共特征,我们可…

文件操作学习总结

磁盘上的⽂件是⽂件。 但是在程序设计中&#xff0c;我们⼀般谈的⽂件有两种&#xff1a; 程序⽂件、数据⽂件 &#xff08;从⽂件功能的⻆度来分类 的&#xff09;。 程序⽂件 &#xff1a; 程序⽂件包括源 程序⽂件&#xff08;后缀为.c&#xff09; , ⽬标⽂件&#xff0…

虚幻学习笔记16—C++和3DUI(二)

一、前言 上一篇虚幻学习笔记15—C和UI&#xff08;一&#xff09;中介绍了通过C代码创建2D的ui&#xff0c;本章主要讲解怎么用C代码创建3D的UI&#xff0c;在虚幻学习笔记3—UI跟随弹窗这章中讲解了怎样用蓝图创建一个3D的UI&#xff0c;并且和其交互。 本系列使用的虚幻5.2.…

Hive参数操作和运行方式

Hive参数操作和运行方式 1、Hive参数操作 1、hive参数介绍 ​ hive当中的参数、变量都是以命名空间开头的&#xff0c;详情如下表所示&#xff1a; 命名空间读写权限含义hiveconf可读写hive-site.xml当中的各配置变量例&#xff1a;hive --hiveconf hive.cli.print.headert…

abpvnext框架的项目部署到linux arm64版的docker中

参考&#xff1a; windows10下安装的docker 导出镜像到另一个电脑_docker镜像拷贝另一台机器的镜像-CSDN博客 前提条件&#xff1a; 1、vs2022&#xff0c;我的电脑本机安装有windows版docker desktop 。 2、linux中已经安装好docker&#xff0c;安装了sftp。这部分可以自行…

vue proxy代理 和 Nginx 配置跨域

vue.config.js文件中配置的代理&#xff1a; devServer: {port: 9095,// open: true, // 配置项目在启动时自动在浏览器打开proxy: {/yh: { // /api是代理标识&#xff0c;一般是每个接口前的相同部分target: "http://192.168.5.58:8002", // 请求地址&#xff0c;一…

2023年12月GESP认证Scratch图形化等级考试(四级)真题试卷

2023年12月GESP认证Scratch图形化等级考试&#xff08;四级&#xff09;真题试卷 题目总数&#xff1a;27 总分数&#xff1a;100 选择题 第 1 题 单选题 现代计算机是指电子计算机&#xff0c;它所基于的是&#xff08; &#xff09;体系结构 A. 艾伦图灵 B. …

Kafka为什么能高效读写数据

1&#xff09;Kafka 本身是分布式集群&#xff0c;可以采用分区技术&#xff0c;并行度高&#xff08;生产消费方并行度高&#xff09;&#xff1b; 2&#xff09;读数据采用稀疏索引&#xff0c;可以快速定位要消费的数据&#xff1b; 3&#xff09;顺序写磁盘&#xff1b; …

vue3项目引入电子签名(可横屏竖屏)

实现效果&#xff1a;&#xff08;左边横屏&#xff0c;右边竖屏&#xff09; 前言&#xff1a;【使用开源项目smooth-signature 实现签名的功能。Gitee 地址是 &#xff1a;GitHub - linjc/smooth-signature: H5带笔锋手写签名&#xff0c;支持PC端和移动端&#xff0c;任何前…

局域网其他pc如何访问宿主机虚拟机IP?

文章目录 背景贝瑞蒲公英设置虚拟机网络连接测试 背景 使用贝瑞蒲公英异地组网&#xff0c;将家里的pc作为pgsql服务器在公司使用&#xff0c;但是虚拟机的ip和端口访问不了 贝瑞蒲公英 设置虚拟机网络 就是添加端口转发规则 连接测试 公网内其他pc连接测试 可以看到已经连接成…

【python基础】-- yarn add 添加依赖的各种类型

目录 1、安装 yarn 1.1 使用npm安装 1.2 查看版本 1.3 yarn 淘宝源配置 2、安装命令说明 2.1 yarn add&#xff08;会更新package.json和yarn.lock&#xff09; 2.2 yarn install 2.3 一些操作 2.3.1 发布包 2.3.2 移除一个包 2.3.3 更新一个依赖 2.3.4 运行脚本 …

知乎上高频提问:Redis到底是单线程还是多线程程序?

1.概述 这里我们先给出问题的全面回答&#xff1a;Redis到底是多线程还是单线程程序要看是针对哪个功能而言&#xff0c;对于核心业务功能部分(命令操作处理数据)&#xff0c;Redis是单线程的&#xff0c;主要是指 Redis 的网络 IO 和键值对读写是由一个线程来完成的&#xff…

Potplayer播放器远程访问群晖WebDav本地资源【内网穿透】

文章目录 本教程解决的问题是&#xff1a;按照本教程方法操作后&#xff0c;达到的效果是&#xff1a;1 使用环境要求&#xff1a;2 配置webdav3 测试局域网使用potplayer访问webdav3 内网穿透&#xff0c;映射至公网4 使用固定地址在potplayer访问webdav 国内流媒体平台的内容…

node.js mongoose aggregate

目录 官方文档 简述 Aggregate的原型方法 aggregate进行操作 官方文档 Mongoose v8.0.3: Aggregate 简述 在 Mongoose 中&#xff0c;Aggregate 是用于执行 MongoDB 聚合操作的类。MongoDB 聚合操作是一种强大的数据处理工具&#xff0c;可以用于对集合中的文档进行变换和…

智慧安防视频监控EasyCVR如何通过回调接口向第三方平台推送RTSP视频通道离线通知

安防视频监控系统EasyCVR能在局域网、公网、专网等复杂的网络环境中部署&#xff0c;可支持4G、5G、WiFi、有线等方式进行视频的接入与传输、处理和分发。平台能将接入的视频流进行汇聚、转码、多格式输出和分发&#xff0c;具体包括&#xff1a;RTMP、RTSP、HTTP-FLV、WebSock…