C语言 宏

目录

一、宏定义

1.1 预定义符号

1.2 预处理指令 #define

1.3 带有副作用宏定义

1.4 宏和函数的一个对比

​编辑

1.5 #undef

二、条件编译

2.1 #if、#else、#elif、#endif

2.2 #ifdef和#ifndef

2.3 C语言中如何通过条件编译来预防头文件的重复包含?


一、宏定义

在C语言中,宏定义是通过  #define 关键字实现的,它可以将被定义的标识符替换为相应的字符串或代码片段。宏定义主要用于简化代码、提高程序的通用性和易读性,同时也能在一定程度上提高程序的运行效率

1.1 预定义符号

在C语言中,预定义符号是由编译器提供的,它们具有特殊的含义和功能。

以下是C语言中的一些常见预定义符号及其用途:

  • __FILE__:表示当前源文件的文件名,可用于调试时显示文件名。 
  • __LINE__:表示当前代码所在的行号,可用于调试时显示代码位置。 
  • __DATE__:表示当前编译的日期,格式为"MMM DD YYYY",例如"Jul 29 2023"。 
  • __TIME__:表示当前编译的时间,格式为"HH:MM:SS",例如"10:30:36"。 
  • __STDC__:如果编译器遵循ANSI C标准,其值为1,否则未定义。这个符号通常用于检测编译器的兼容性。

在VS中(__STDC__)未定义,所以该编译器不支持ANSC l 标准,如下图:

1.2 预处理指令 #define

宏定义主要有两种类型:

  • 不带参数的宏定义和。
  • 带参数的宏定义。

不带参数的宏定义简单地将宏名替换为定义的字符串,例如:

#define PI 3.14159

带参数的宏定义除了进行简单的文本替换外,还会对参数进行计算。例如:

#define S(a,b) a*b

#define 定义的规则

  • 宏替换发生在预处理阶段,即编译前
  • 宏替换是简单的文本替换,不涉及计算
  • 宏替换不占用运行时间,只占用编译时间

1.3 # 和 ##

#操作符用于预编译时期,将宏参数转换为字符串。如图:

##操作符用于在预编译期间将两个宏参数连接起来,形成一个单独的标识符。如图:

1.3 带有副作用宏定义

例如,假设有一个宏定义MAX(a, b),它在预处理阶段展开为((a) > (b) ? (a) : (b))。如果使用MAX(x++, y++),预处理后的结果将是((x++) > (y++) ? (x++) : (y++))。这意味着xy都将被自增两次,这可能并非程序员原本的意图。

正常情况下:

带有副作用情况下:

  • ret 返回的是7,自增了一次,在问好(?)前面。
  • a 的值是3 ,自增了一次,在问号(?)前面。
  • b 的值是8,自增了两次,在则问号前后各增加一次。
1.4 宏和函数的一个对比

相比之下,宏在预编译阶段将宏名替换为后面的替换体,避免了函数调用的开销。宏的展开是简单的文本替换,没有类型检查,因此可能会导致一些潜在的错误。然而,宏的使用可以提高代码的可读性和可维护性,并且在某些情况下,宏的使用可以提高程序的效率。

1.5 #undef

#undef是C语言预处理指令的一种,它的作用是取消之前通过#define指令定义的宏或符号常量

#include <stdio.h>

int main() {
    #define MAX 200
    printf("MAX = %d\n", MAX);
    #undef MAX
    int MAX = 10;
    printf("MAX = %d\n", MAX);
    return 0;
}

二、条件编译

条件编译的主要指令包括#if#else#elif#endif#ifdef#ifndef

2.1 #if#else#elif#endif
  • #if用于判断表达式是否为真,如果为真则编译其后的代码;
  • #else用于与#if搭配,当前面的条件不满足时,编译#else后的代码;
  • #elif相当于#else#if的结合,用于多个条件中的其他条件
  • #endif用于结束一个条件编译块;
2.2 #ifdef#ifndef

#ifndef#ifdef相反,用于判断一个宏是否未被定义,如果未被定义则编译其后的代码。

2.3 C语言中如何通过条件编译来预防头文件的重复包含?

1、使用#ifndef、#define、#endif预处理指令

#ifndef MY_HEADER_H
#define MY_HEADER_H
// 头文件内容
#endif

当头文件第一次被包含时,由于没有定义MY_HEADER_H,编译器会执行#define指令,定义该宏。当头文件再次被包含时,由于该宏已被定义,编译器会跳过#ifndef#endif之间的代码,从而避免重复包含。

2、使用#pragma once指令:

#pragma once
// 头文件内容

这种方法的好处是不会出现宏名冲突的问题,且对于大型项目来说,可以提高编译速度。 

评论 14
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值