c语言预处理和程序环境

ZHENGZJM ZHENGZJM     2023-04-09     217

关键词:

目录

程序的环境

运行环境 

翻译环境

编译的过程 

预编译阶段

编译阶段 

汇编阶段 

链接阶段(不属于编译阶段)

预处理详解

预定义符号

 #define

#define定义标识符

#define定义宏

#define的替换规则 

#和##的使用 

带副作用的宏参数 

宏和函数的对比

一些命名的规则 

#undef

命令行的定义

条件编译

常见的条件编译指令 

 文件包含

总结 


 

程序的环境

在ANSI C的任何一种的实现中,都存在则两个环境。

一个是翻译环境,在这个环境中,源代码被转化为可执行的机器指令。(通过编译器和连接器来进行转换)。

一个是执行环境,它用于实际执行代码。

运行环境 

程序必须载入内存中,在有操作系统的环境中:一般这个由操作系统完成。在独立环境中,程序的载入必须由手工安排,也可能是通过可执行代码置入只读内存来完成。

翻译环境

 

每一个源程序都是单独编译转换成目标文件,每个目标文件通过连接器捆绑在一起,从而形成一个单一而完整的可执行程序 。

同时,连接器会引入标准c函数库中任何被改程序所用到的函数,而且他可以搜索程序员个人的程序库,将其需要的函数也连接到程序中。

编译的过程 

编译的过程实际上有三个过程:预编译阶段、编译阶段、汇编阶段。

预编译阶段

 主要是处理一些预处理指令。

编译阶段 

编译阶段包含四个部分:语法分析、词法分析、语义分析、符号汇总。将代码转换成汇编代码。

其中,对于符号汇总:将全局型的符号收集起来。

汇编阶段 

形成符号表、将汇编指令转换成二进制机器指令。 

链接阶段(不属于编译阶段)

合并段表,符号表的合并和符号表的重定位。

预处理详解

对于预处理,我们需要了解预定义符号、#define的相关知识、#和##的使用等知识。 

预定义符号

__FILE__ :进行编译的源文件%s

__LINE__:文件当前的行号%d

__DATA__:文件被编译的日期%s

__TIME__:文件被编译的时间%s

__STDC__ :如果编译器遵循ANSI C,其值为1,否则未定义。

这些预定义符号都是语言内置的。

 #define

#define定义标识符

语法形式:#define name stuff。实际上就是将name 替换成stuff。

#define定义宏

#define 机制包括了一个规定,允许把参数替换到文本当中,这种实现通常称为宏。

宏的申明方式:#define name(parament-list) stuff。

定义宏时要注意的点:参数列表的左括号必须和name紧挨着,要不然会被解析成stuff的一部分。 在定义宏时,我们尽量将参数都带上括号,并且将整个宏用括号括起来,防止因为符号优先级的问题造成我们不想看见的结果。

#define的替换规则 

1、在调用宏时,我们首先要对参数进行检查,看看是否包含任何由#define定义的标识符,如果有,它们将首先被替换。

2、替换文本随后被插入到程序中原来文本的位置,对于宏,参数名将被它们的值所替换。

3、最后,再次对文本进行扫描,观察是否还存在#define定义的符号,如果有,重复上面的过程。

4、宏不能出现递归。

5、当预处理器搜索#define定义的符号时,字符串常量的内容并不被搜索。

#和##的使用 

#的作用是将一个宏参数转变成对应的字符串。

##的作用是将位于他两边的符号合并成一个符号,它允许宏定义从分离的文本片段创建标识符。

带副作用的宏参数 

 当宏参数在宏定义中出现不止一次时,如果参数带有副作用,那么在使用这个宏时没有看产生不可预测的后果。  

x+1不带副作用

x==带副作用

宏和函数的对比

看完上面讲的,你也许会疑惑,宏和函数好像差不多啊,有什么区别呢?

宏通常被用在执行简单的运算之中,其原因是:用于调用函数和从函数返回的代码可能比实际执行这个小型计算工作所需要的时间更多。函数的参数要求是必须声明为特定类型,所以函数只能在类型合适的表达式上使用,而宏可以适用于不同类型(因为只是替换罢了)。

所以宏比函数在程序的规模和速度方面更胜一筹。 

+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

但宏也有缺点:1、每次使用宏时,以分红定义的代码将插入到程序中。除非宏比较短、否则可能大幅度增加程序的长度。2、宏是没办法调试的(在预处理阶段就把文本给替换掉了)。

3、宏不够严谨。4、宏可能会带来运算符优先级的问题,导致程序容易出错。

属性#define定义宏函数
代码长度每次使用时,宏代码都会被插入到程序中。出来非常小的宏之外,程序的长度会大幅度增长函数代码只出现于一个地方,每次使用这个函数时,都调用同一个地方的同一份代码。
执行速度更快存在函数的调用和返回的额外开销,所以相对慢一些。
操作符优先级宏参数的求值是在所有周围表达式的上下文环境里,除非加上括号,否则临近操作符的优先级可能会产生不可预料的后果,所以建议在宏书写的时候多加些括号。函数参数值在函数调用的时候求值一次,它的结果值传递给函数。表达式的求值结果更容易预测。
带有副作用的参数参数可能被替换到宏体中的任意位置,所以带有副作用的参数求值可能会产生不可预料的结果函数参数只在传参的时候求值一次,结果更容易被控制
参数类型宏的参数和类型无关,只要对参数的操作是合法的,他就可以使用任何类型的参数函数的参数是与类型有关的,如果参数的类型不同,就需要不同的函数,即使他们执行的内容是相同的(比如算整数相加和算浮点数相加)
调试宏不能被调试函数是可以逐语句调试的
递归宏不能递归函数是可以递归的

 

一些命名的规则 

我们一般将宏的名字全部大写,函数名不要全部大写。

#undef

这条指令用于移除一个宏定义 

#undef name

如果一个现存的一个名字需要被重新定义,那么它的旧名字首先要被移除。

命令行的定义

 许多c的编译器提供了一种方法:允许在命令行中定义符号(在启动编译的过程中)

比如定义一个数组int arr【arr_size】;我们可以先不定义arr_size,在编译过程中使用命令上进行定义。

条件编译

如果我们使用条件编译指令,那么在编译一个程序的时候,我们如果要将一条(或者一组) 编译或者放弃是很方便的。

例如调试行的代码,留着碍事,但删除又很可惜,使用我们可以选择使用选择性的编译。

常见的条件编译指令 

#if 常量表达式

\\\\语句。。

#endif 

多个分支语句的条件编译

#if 常量表达式

//语句

#elif 常量表达式

//语句

#else

//语句

#endif 

判断是否被定义的条件编译

#if defined(symbol) (或者写成#ifdef symbol,这俩是等价的)

#if !defined(symbol)(或者写成#ifndef symbol)

嵌套条件编译指令

#if defined(OS_UNIX)

     #ifdef OPTION1

             unix_version_option1();

     #endif

     #ifdef OPTION2

              unix_version_option2();

     #endif

#elif defined(OS_MSDOS)

      #ifdef OPTION2

              msdos_version_option2();

#endif

 文件包含

头文件的引用一般有两种形式,一种是<>,另外一种是“ ”,前一种多用于库函数的头文件引用,后一种多是引用程序员个人写的头文件。

后一种的查找策略:先在原文件所在的目录下查找,如果目标头文件没找到,编译器就像查找库函数头文件一样在标准位置查找头文件,如还没查找到,则提示编译错误。

在引用头文件时,有时候会因为嵌套引用头文件造成头文件的重复引用,进而造成文件内容的重复,我们常常通过条件编译来解决这个问题。 

总结 

本文介绍了程序环境、编译过程、#define定义的标识符和宏以及相关知识,头文件的应用,预处理的详细解析。

c语言预处理和程序环境

...译阶段 汇编阶段 链接阶段(不属于编译阶段)预处理详解预定义符号 #define#define定义标识符#define定义宏#define的替换规则 #和##的使用 带副作用的宏参数 宏和函数的对比一些命名的规则 #undef命令行的定义条件编译... 查看详情

c语言进阶学习笔记——程序环境和预处理(代码片段)

...翻译环境和执行环境详解编译+链接翻译环境运行环境预处理详解预定义符号#define#define定义标识符#define定义宏#define替换规则#和##带副作用的宏参数宏和函数宏和函数的对比命名约定#undef命令行定义条件编译文件包含头文件被... 查看详情

c语言编译过程,满满的干货!!!(代码片段)

程序环境和预处理一、程序翻译和运行环境二、预处理详解1.预定义符号2.define定义宏3.#和##的区别4.宏和函数好坏比较5.命名约定6.头文件中<>和""区别7.条件编译一、程序翻译和运行环境翻译环境:在翻译环境中ÿ... 查看详情

c语言进阶六.预处理

 (1)程序的翻译环境和执行环境在ANSIC的任何一种实现中,存在两个不同的环境。第1种是翻译环境,在这个环境中源代码被转换为可执行的机器指令。包含编译加链接第2种是执行环境,它用于实际执行代码。(2)详解编译... 查看详情

c语言中程序的编译(预处理操作)+链接详解(详细介绍程序预编译过程)(代码片段)

...目录1.前言2.翻译环境和运行环境2.1翻译环境2.2运行环境3.预处理详解3.1预定义符号3.2#define定义的标识符常量和宏3.2.1#define定义的标识符常量3.2.2#define定义的宏3.2.3#define替换规则3.2.4#和##3.2.5带副作用的宏参数3.3宏和函数的对比4.... 查看详情

c语言-程序环境和预处理(代码片段)

文章目录预处理详解1.预定义符号2.#define2.1#define定义的标识符2.2#define定义宏2.3#define替换规则注意事项:2.4#和###的作用##的作用2.5带副作用的宏参数2.6宏和函数的对比宏的优势:宏的劣势:宏和函数的一个对比命名约... 查看详情

梦开始的地方——c语言预处理+编译过程(代码片段)

文章目录C语言程序的编译(预处理)1.编译和链接1)编译的几个阶段预编译阶段编译阶段汇编阶段2)链接2.预处理1)预定义符号2)#define3)#和##4)带副作用的宏参数5)宏和函数对比3.常见预处理命令1)#undef2)命令行定义3)条件编译4)文件包含5... 查看详情

c语言篇——程序的编译(代码片段)

...翻译环境和执行环境编译和链接翻译环境编译的几个阶段预处理编译汇编链接运行环境程序的翻译环境和执行环境在ANSIC的任何一种实现中,存在两个不同的环境:第1种是翻译环境,在这个环境中源代码被转换为可执行的... 查看详情

c语言篇——程序的编译(代码片段)

...翻译环境和执行环境编译和链接翻译环境编译的几个阶段预处理编译汇编链接运行环境程序的翻译环境和执行环境在ANSIC的任何一种实现中,存在两个不同的环境:第1种是翻译环境,在这个环境中源代码被转换为可执行的... 查看详情

c语言程序的翻译环境和执行环境

查看详情

程序环境和预处理(代码片段)

...a6;翻译环境(编译+链接)💦运行环境二、预处理详解💦预定义符号💦#define定义标识符💦#define定义宏💦#define替换规则💦#和##(奇怪的用法)💦带副作用的宏参数💦宏和函数的对比... 查看详情

程序的环境和预处理

1.程序的环境预编译1.头文件的包含2.#define的预处理指令的执行3.注释的删除编译汇编链接运行1.程序的环境写过无数代码的你是否想过你写的.c文件编译链接运行形成.exe文件的中间是怎样执行的吗?从.c文件到.exe文件到代码... 查看详情

c语言和c++有啥区别?

一、主体不同1、C语言:是一门面向过程的、抽象化的通用程序设计语言,广泛应用于底层开发。2、C++:是C语言的继承,它既可以进行C语言的过程化程序设计,又可以进行以抽象数据类型为特点的基于对象的程序设计。二、优... 查看详情

c语言进阶学习笔记七程序执行+调试技巧(实用技巧篇)(代码片段)

文章目录一、程序执行篇①预处理详解②宏定义③define标识符字符串④define宏名(参数表)字符串⑤宏和函数对比二、调试技巧篇①什么是bug?②调试是什么?有多重要?③debug和release的介绍④windows环境调试介绍⑥如何写出... 查看详情

vb语言和c语言有啥区别?

一、主体不同1、vb语言:是Microsoft公司开发的一种通用的基于对象的程序设计语言。2、c语言:是一门面向过程的、抽象化的通用程序设计语言,广泛应用于底层开发。二、特点不同1、vb语言:结构化的、模块化的、面向对象的... 查看详情

刨析c语言的程序环境

C语言程序运行的环境1、程序的翻译环境和执行环境2、详解编译+链接编译过程链接过程翻译过程详解3、运行环境1、程序的翻译环境和执行环境在ANSIC的任何一种实现中,存在两个不同的环境。第1种是翻译环境,在这... 查看详情

刨析c语言的程序环境

C语言程序运行的环境1、程序的翻译环境和执行环境2、详解编译+链接编译过程链接过程翻译过程详解3、运行环境1、程序的翻译环境和执行环境在ANSIC的任何一种实现中,存在两个不同的环境。第1种是翻译环境,在这... 查看详情

w1-3-2程序的开发和执行及其支撑环境

...红色为操作,黑色为生成的结果源程序(hello.c)[文本]-->预处理【cpp】-->源程序(hello.i)[文本]-->编译(cc1)-->汇编语言程序(hello.s)[文本]-->汇编(as)-->可重定位目标程序(hello.o)[二进制]+printf.o-->链接(ld)-->可执行目 查看详情