当前位置: 首页 > news >正文

一个CC++程序的生命历程

翻了好多博客,内容星星点点,没找到我想要的,现在吸取大神精华,加上本人拙见,总结如下:

一个C或C++程序从你开始编写,到结束,整个过程,都做了些什么,请看下文:

先看大体的过程:看图:


我在这里主要想说的就是,程序到这每个阶段都干了什么,这个有时候很重要,举个例子:

#define BSC //
#define BMC /*
#define EMC */

//1.双斜杠注释
BSC lining is a goog boy

//2./**/注释
BMC lining is good boy EMC
上边的宏替换,很容易看懂,但是,这确实是错的,原因也很简单,注释先于预处理指令处理,而宏替换又是在预处理的时候处理,当宏取替换的时候,注释早被处理了,那么问题就来了,我们该如何知道,各种处理过程都干了什么,这问题又回到了开始,现在来分析:

——》预处理
包括:

#define
#error
#include
#if
#else
#elif
#endif
#ifdef
#ifndef
#undef
#line
#pragma

所有预处理命令均以符号#开头。

一 #define

命令#define定义了一个标识符及一个串。在源程序中每次遇到该标识符时,均以定义的串代换它。ANSI标准将标识符定义为宏名,将替换过程称为宏替换。命令的一般形式为:

#define identifier string

分析:

1该语句没有分号。在标识符和串之间可以有任意个空格,串一旦开始,仅由一新行结束。

2宏名定义后,即可成为其它宏名定义中的一部分。

3 宏替换仅仅是以文本串代替宏标识符,前提是宏标识符必须独立的识别出来,否则不进行替换。例如:

#define XYZ this is a tes

使用宏printf("XYZ");//该段不打印"this is a test"而打印"XYZ"。因为预编译器识别出的是"XYZ"

4如果串长于一行,可以在该行末尾用一反斜杠' \'续行。

#defineLONG_STRING"this is a very long\
string that is used as an example"

5 C语言程序普遍使用大写字母定义标识符。

6 用宏代换代替实在的函数的一大好处是宏替换增加了代码的速度,因为不存在函数调用的开销。但增加速度也有代价:由于重复编码而增加了程序长度。
 
二 #error

命令#error强迫编译程序停止编译,用于程序调试

#error指令使预处理器发出一条错误消息,该消息包含指令中的文本.这条指令的目的就是在程序崩溃之前能够给出一定的信息。

三 #include

命令#i nclude使编译程序将另一源文件嵌入带有#include的源文件,被读入的源文件必须用双引号或尖括号括起来。例如:

#include"stdio.h"或者#include

这两行代码均使用C编译程序读入并编译用于处理磁盘文件库的子程序。

将文件嵌入#i nclude命令中的文件内是可行的,这种方式称为嵌套的嵌入文件,嵌套层次依赖于具体实现。

如果显式路径名为文件标识符的一部分,则仅在那些子目录中搜索被嵌入文件。否则,如果文件名用双引号括起来,则首先检索当前工作目录。如果未发现文件,则在命令行中明

的所有目录中搜索。如果仍未发现文件,则搜索实现时定义的标准目录。

如果没有显式路径名且文件名被尖括号括起来,则首先在编译命令行中的目录内检索。如果文件没找到,则检索标准目录,不检索当前工作目录。

四 条件编译命令

有几个命令可对程序源代码的各部分有选择地进行编译,该过程称为条件编译。商业软件公司广泛应用条件编译来提供和维护某一程序的许多顾客版本

#if、#else,#elif及#endif

#if的一般含义是如果#if后面的常量表达式为true,则编译它与#endif之间的代码,否则跳过这些代码。命令#endif标识一个#if块的结束。

#if constant-expression
statement sequence
#endif
Eg:
#define MAX 91
#include
using namespace std;
int main()
{
#if MAX > 99
       cout<<"MAX is bigger than 99"<
#elif MAX > 90
       cout<<"MAX is bigger than 90"<
#else
       cout<<"MAX is smaller than 90"<
#endif
       return 0;
}

跟在#if后面的表达式在编译时求值,因此它必须仅含常量及已定义过的标识符,不可使用变量。表达式不许含有操作符sizeof(sizeof也是编译时求值)。
 
#else命令的功能有点象C语言中的else;#else建立另一选择(在#if失败的情况下)。注意,#else属于#if块。
 
#elif命令意义与ELSE IF 相同,它形成一个if else-if阶梯状语句,可进行多种编译选择。#elif 后跟一个常量表达式。如果表达式为true,则编译其后的代码块,不对其它#elif表达式

进行测试。否则,顺序测试下一块。

#if expression
statement sequence
#elif expression1
statement sequence
#endif<span style="font-family: Arial, Helvetica, sans-serif; background-color: rgb(255, 255, 255);"> </span>
在嵌套的条件编译中#endif、#else或#elif与最近#if或#elif匹配。

# ifdef 和# ifndef
条件编译的另一种方法是用#ifdef与#ifndef命令,它们分别表示"如果有定义"及"如果无定义"。# ifdef的一般形式是:
# ifdef macroname
statement sequence
#endif
#ifdef与#ifndef可以用于#if、#else,#elif语句中,但必须与一个#endif。
#define MAX 91
#include
using namespace std;
 
int main()
{
#ifdef MAX
       cout<<"hello,MAX!"<
#else
       cout<<"where is MAX?"<
#endif
#ifndef LEO
       cout<<"LEO is not defined"<
#endif
       return 0;
}
命令#undef 取消其后那个前面已定义过有宏名定义。一般形式为:
#undef macroname
命令#line改变__LINE__与__FILE__的内容,它们是在编译程序中预先定义的标识符。命令的基本形式如下:
#line number["filename"]
其中的数字为任何正整数,可选的文件名为任意有效文件标识符。行号为源程序中当前行号,文件名为源文件的名字。命令#line主要用于调试及其它特殊应用。注意:在#line后面

的数字标识从下一行开始的数字标识。

#line 100 "jia"
       cout<<"#line change line and filename!"<
       cout<<__LINE__<
       cout<<__FILE__<
五 #pragma

命令#pragma 为实现时定义的命令,它允许向编译程序传送各种指令。

#pragma的作用是设定编译器的状态或者是指示编译器完成一些特定的动作。#pragma指令对每个编译器给出了一个方法,在保持与C和C++语言完全兼容的情况下,给出主机或操作

系统专有的特征。依据定义,编译指示是机器或操作系统专有的,且对于每个编译器都是不同的。

其格式一般为: 

#Pragma Para

1 message 参数。

 Message 参数能够在编译信息输出窗口中输出相应的信息,这对于源代码信息的控制是非常重要的。其使用方法为:

#pragma message(“消息文本”)

当编译器遇到这条指令时就在编译输出窗口中将消息文本打印出来。

当我们在程序中定义了许多宏来控制源代码版本的时候,我们自己有可能都会忘记有没有正确的设置这些宏,此时我们可以用这条指令在编译的时候就进行检查。假设我们希望判

断自己有没有在源代码的什么地方定义了_X86这个宏可以用下面的方法

#ifdef _X86
#pragma message(“_X86 macro activated!”)
#endif
当我们定义了_X86这个宏以后,应用程序在编译时就会在编译输出窗口里显示“_
X86 macro activated!”
2  code_seg 参数。

格式如:

#pragma code_seg( ["section-name"[,"section-class"] ] )

它能够设置程序中函数代码存放的代码段,当我们开发驱动程序的时候就会使用到它。


3 #pragma once (比较常用)
只要在头文件的最开始加入这条指令就能够保证头文件被编译一次。这条指令实际上在VC6中就已经有了,但是考虑到兼容性并没有太多的使用它。

4 #pragma hdrstop
表示预编译头文件到此为止,后面的头文件不进行预编译。BCB可以预编译头文件以加快链接的速度,但如果所有头文件都进行预编译又可能占太多磁盘空间,所以使用这个选项

排除一些头文件

有时单元之间有依赖关系,比如单元A依赖单元B,所以单元B要先于单元A编译。你可以用#pragma startup指定编译优先级,如果使用了#pragma package(smart_init) ,BCB就会

根据优先级的大小先后编译。

5 #pragma resource "*.dfm"

表示把*.dfm文件中的资源加入工程。*.dfm中包括窗体外观的定义。

6 #pragma warning( disable : 4507 34; once : 4385; error : 164 )

等价于:

#pragma warning(disable:4507 34) /* 不显示4507和34号警告信息。如果编译时总是出现4507号警告和34号警告,  而认为肯定不会有错误,可以使用这条指令。*/
#pragma warning(once:4385) // 4385号警告信息仅报告一次
#pragma warning(error:164) // 把164号警告信息作为一个错误。
同时这个pragma warning 也支持如下格式:
#pragma warning( push [ ,n ] )
#pragma warning( pop )
这里n代表一个警告等级(1---4)。
#pragma warning( push )保存所有警告信息的现有的警告状态。
#pragma warning( push, n)保存所有警告信息的现有的警告状态,并且把全局警告等级设定为n。
#pragma warning( pop )向栈中弹出最后一个警告信息,在入栈和出栈之间所作的一切改动取消。例如:
#pragma warning( push )
#pragma warning( disable : 4705 )
#pragma warning( disable : 4706 )
#pragma warning( disable : 4707 )
//.......
#pragma warning( pop )
在这段代码的最后,重新保存所有的警告信息(包括4705,4706和4707)。

7 pragma comment(...)

该指令将一个注释记录放入一个对象文件或可执行文件中

常用的lib关键字,可以帮我们连入一个库文件。 

8 progma pack(n)

指定结构体对齐方式。#pragma pack(n)来设定变量以n字节对齐方式。

n 字节对齐就是说变量存放的起始地址的偏移量有两种情况:

第一、如果n大于等于该变量所占用的字节数,那么偏移量必须满足默认的对齐方式

第二、如果n小于该变量的类型所占用的字节数,那么偏移量为n的倍数,不用满足默认的对齐方式。

结构的总大小也有个约束条件,分下面两种情况:如果n大于所有成员变量类型所占用的字节数,那么结构的总大小必须为占用空间最大的变量占用的空间数的倍数; 否则必须为n的倍数。


声明:部分转载,希望对你有用;

———》编译器:

经过预编译得到的输出文件中,将只有常量,如数字、字符串、变量的定义,以及C语言的关键字,如main, if, else, for, while, {, }, +, -, *, \, 等等。

预编译程序所要做的工作就是通过词法分析和语法分析,在确认所有的指令都符合语法规则之后,将其翻译成等价的中间代码表示或汇编代码。

———》汇编

汇编过程实际上指把汇编语言代码翻译成目标机器指令的过程。对于被翻译系统处理的每一个C语言源程序,都将最终经过这一处理而得到相应的目标文件。

目标文件中所存放的也就是与源程序等效的目标的机器语言代码。

———》链接程序

由汇编程序生成的目标文件并不能立即就被执行,其中可能还有许多没有解决的问题。

例如,某个源文件中的函数可能引用了另一个源文件中定义的某个符号(如变量或者函数调用等);在程序中可能调用了某个库文件中的函数,等等。

所有的这些问题,都需要经链接程序的处理方能得以解决。链接程序的主要工作就是将有关的目标文件彼此相连接,也即将在一个文件中引用的符号同该符号在另外一个文件中的

定义连接起来

如果后期还有发现,会续上;

赐教!


转载于:https://www.cnblogs.com/melons/p/5791838.html

相关文章:

  • 为什么要有ID发号器、原理是什么以及如何实现?
  • Python系列之模块、和字符串格式化
  • 分布式之数据库和缓存双写一致性方案解析!
  • linux查看文件内容的常见命令
  • 慢SQL!压垮团队的最后一根稻草!
  • 2017年秋招美团Java程序员开发,看我如何拿到offer
  • Javascript中常用事件的命名
  • 阿里的面试官都喜欢问哪些问题?
  • 浅谈C语言中结构体的初始化
  • Spring AOP中的JDK和CGLib动态代理哪个效率更高?
  • 2016百度之星 - 初赛(Astar Round2A)
  • 为什么需要分布式配置中心?
  • 线上出故障了!我慌得一匹!教大家如何应对在线故障!
  • mysql远程访问cannot connect(10038) 问题解决的过程
  • Spring Cloud技术栈还没有学完!Hystrix又双叒叕停止更新了!
  • [rust! #004] [译] Rust 的内置 Traits, 使用场景, 方式, 和原因
  • Bytom交易说明(账户管理模式)
  • Create React App 使用
  • JavaScript 是如何工作的:WebRTC 和对等网络的机制!
  • PHP的Ev教程三(Periodic watcher)
  • PHP的类修饰符与访问修饰符
  • SwizzleMethod 黑魔法
  • vue.js框架原理浅析
  • 给Prometheus造假数据的方法
  • 深入浏览器事件循环的本质
  • 我与Jetbrains的这些年
  • 用Python写一份独特的元宵节祝福
  • 扩展资源服务器解决oauth2 性能瓶颈
  • ​520就是要宠粉,你的心头书我买单
  • #控制台大学课堂点名问题_课堂随机点名
  • #每天一道面试题# 什么是MySQL的回表查询
  • #我与Java虚拟机的故事#连载16:打开Java世界大门的钥匙
  • (BFS)hdoj2377-Bus Pass
  • (pojstep1.1.1)poj 1298(直叙式模拟)
  • (pytorch进阶之路)CLIP模型 实现图像多模态检索任务
  • (利用IDEA+Maven)定制属于自己的jar包
  • (亲测成功)在centos7.5上安装kvm,通过VNC远程连接并创建多台ubuntu虚拟机(ubuntu server版本)...
  • (三) diretfbrc详解
  • (原創) 如何刪除Windows Live Writer留在本機的文章? (Web) (Windows Live Writer)
  • (转) RFS+AutoItLibrary测试web对话框
  • (转)如何上传第三方jar包至Maven私服让maven项目可以使用第三方jar包
  • (转)项目管理杂谈-我所期望的新人
  • .NET Standard 支持的 .NET Framework 和 .NET Core
  • .Net 访问电子邮箱-LumiSoft.Net,好用
  • .NET 设计模式初探
  • ::前边啥也没有
  • @Async注解的坑,小心
  • @Valid和@NotNull字段校验使用
  • [2018-01-08] Python强化周的第一天
  • [BSGS算法]纯水斐波那契数列
  • [C++]打开新世界的大门之C++入门
  • [Contiki系列论文之2]WSN的自适应通信架构
  • [JS]变量
  • [LeetBook]【学习日记】数组内乘积
  • [LeetCode] 93. Restore IP Addresses 复原IP地址