热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

C语言三)预处理指令

一、预处理指令1.定义C语言在对源程序进行编译之前,会先对一些特殊的预处理指令做解释,如:#include,产生一个新的源程序,这个过程称为“编译预处理”,之后再进行通常的编译

一、预处理指令

1.定义

C语言在对源程序进行编译之前,会先对一些特殊的预处理指令做解释,如:#include,产生一个新的源程序,这个过程称为“编译预处理”,之后再进行通常的编译工作。

2.作用范围

预处理指令可以出现在程序的任意位置,它的作用范围是从它出现的位置到文件末尾。习惯上,我们把预处理指令写在源程序开头。

3.类型

C语言提供了预处理指令主要有:宏定义、文件包含和条件编译。


二、不带参数的宏定义

1.一般形式

#define 宏名 字符串

如:#define A 10;

// 源程序中所有的宏名PI在编译预处理的时候都会被3.14所代替
#define PI 3.14
// 根据圆的半径计radius算周长
float girth(float radius) {
    return 2 * PI *radius;
}
int main ()
{
    float g = girth(2);
    printf("周长为:%f", g);
    return 0;
}


2.作用

在编译预处理的时候,将源程序中所有“宏名”替换成右边的字符串,常用来定义常量。

3.注意

1)宏名一般用大写字母,以便和变量名进行区别。

2)对程序中双引号括起来的字符串内的字符,不能进行宏的替换操作。

#define R 10
int main ()
{
    char *s = "Radio"; //这里有R,但是不会替换
    return 0;
}

3)在进行宏的替换操作时,不做语法的检查,只是简单的替换操作而已。只有在替换完后的下一步编译操作,才会进行语法检查。因此,如果有错误,也只会在编译的时候产生。

4)宏名的有效范围默认是从定义位置到文件末尾,如果想终止的话,可以使用“#undef 宏名“,这样,宏名的有效范围就缩减至终止操作的位置。

5)定义宏时,可以使用已经定义过的宏名。

#define R  3.0
#define PI 3.14
#define L  2*PI*R
#define S  PI*R*R

三、带参数的宏定义

1.一般形式

#define 宏名(参数列表) 字符串

2.作用

在编译预处理的时候,将源程序中的宏名替换成字符串,并将字符串中的参数用源程序中的宏名的实际数值代替。

3.注意

1)宏名和参数列表之间不能有空格。

2)定义宏时,参数列表有"()"括起来,字符串的参数也用"()"括起来

#include 
#define D(a) 2*a
int main ()
{
    int b = D(3+4);
    
    printf("%d", b);
    return 0;
}
//2*3+4
//输出结果:10
#include 
#define D(a) 2*(a)
int main ()
{
    int b = D(3+4);
    
    printf("%d", b);
    return 0;
}
//2*(3+4)
//输出结果:14

3)计算结果也用”()“括起来

#include 
#define Pow(a) (a) * (a)
int main(int argc, const char * argv[]) {
    int b = Pow(10) / Pow(2);
    
    printf("%d", b);
    return 0;
}
//10*10/2*2=100
//输出结果:100
#include 
#define Pow(a) ((a) * (a))
int main(int argc, const char * argv[]) {
    int b = Pow(10) / Pow(2);
    
    printf("%d", b);
    return 0;
}
//(10*10)/(2*2)=25
//输出结果:25


4.宏与函数的区别

虽然带参数的宏定义和函数十分相似,但是还是有本质的区别:

1)宏定义不涉及存储空间的分配、参数类型分配、参数传递、返回值的问题。

2)函数在程序运行时执行,而宏替换在编译预处理时执行。所以带参数的宏比函数效率更高。
















C语言三)预处理指令


推荐阅读
  • 本文探讨了C++编程中理解代码执行期间复杂度的挑战,特别是编译器在程序运行时生成额外指令以确保对象构造、内存管理、类型转换及临时对象创建的安全性。 ... [详细]
  • 本文详细介绍了get和set方法的作用及其在编程中的实现方式,同时探讨了点语法的使用场景。通过具体示例,解释了属性声明与合成存取方法的概念,并补充了相关操作的最佳实践。 ... [详细]
  • Python 内存管理机制详解
    本文深入探讨了Python的内存管理机制,涵盖了垃圾回收、引用计数和内存池机制。通过具体示例和专业解释,帮助读者理解Python如何高效地管理和释放内存资源。 ... [详细]
  • 深入解析动态代理模式:23种设计模式之三
    在设计模式中,动态代理模式是应用最为广泛的一种代理模式。它允许我们在运行时动态创建代理对象,并在调用方法时进行增强处理。本文将详细介绍动态代理的实现机制及其应用场景。 ... [详细]
  • 通常情况下,修改my.cnf配置文件后需要重启MySQL服务才能使新参数生效。然而,通过特定命令可以在不重启服务的情况下实现配置的即时更新。本文将详细介绍如何在线调整MySQL配置,并验证其有效性。 ... [详细]
  • 本题要求在一组数中反复取出两个数相加,并将结果放回数组中,最终求出最小的总加法代价。这是一个经典的哈夫曼编码问题,利用贪心算法可以有效地解决。 ... [详细]
  • ListView简单使用
    先上效果:主要实现了Listview的绑定和点击事件。项目资源结构如下:先创建一个动物类,用来装载数据:Animal类如下:packagecom.example.simplelis ... [详细]
  • 深入剖析JVM垃圾回收机制
    本文详细探讨了Java虚拟机(JVM)中的垃圾回收机制,包括其意义、对象判定方法、引用类型、常见垃圾收集算法以及各种垃圾收集器的特点和工作原理。通过理解这些内容,开发人员可以更好地优化内存管理和程序性能。 ... [详细]
  • 本文详细探讨了JavaScript中的作用域链和闭包机制,解释了它们的工作原理及其在实际编程中的应用。通过具体的代码示例,帮助读者更好地理解和掌握这些概念。 ... [详细]
  • Hadoop发行版本选择指南:技术解析与应用实践
    本文详细介绍了Hadoop的不同发行版本及其特点,帮助读者根据实际需求选择最合适的Hadoop版本。内容涵盖Apache Hadoop、Cloudera CDH等主流版本的特性及应用场景。 ... [详细]
  • Appium + Java 自动化测试中处理页面空白区域点击问题
    在进行移动应用自动化测试时,有时会遇到某些页面没有返回按钮,只能通过点击空白区域返回的情况。本文将探讨如何在Appium + Java环境中有效解决此类问题,并提供详细的解决方案。 ... [详细]
  • 本题来自WC2014,题目编号为BZOJ3435、洛谷P3920和UOJ55。该问题描述了一棵不断生长的带权树及其节点上小精灵之间的友谊关系,要求实时计算每次新增节点后树上所有可能的朋友对数。 ... [详细]
  • 深入解析 Android IPC 中的 Messenger 机制
    本文详细介绍了 Android 中基于消息传递的进程间通信(IPC)机制——Messenger。通过实例和源码分析,帮助开发者更好地理解和使用这一高效的通信工具。 ... [详细]
  • Python自动化测试入门:Selenium环境搭建
    本文详细介绍如何在Python环境中安装和配置Selenium,包括开发工具PyCharm的安装、Python环境的设置以及Selenium包的安装方法。此外,还提供了编写和运行第一个自动化测试脚本的步骤。 ... [详细]
  • 本文介绍了如何通过Java代码计算一个整数的位数,并展示了多个基础编程示例,包括求和、平均分计算、条件判断等。 ... [详细]
author-avatar
廖汉林1026_843
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有