【C初阶】预处理

简介: 【C初阶】预处理

前言:在本文中,我将系统的整理一下C语言关于预处理部分的语法,便于整理与回归。

1.预定义符号

在C语言中,C标准提供里一些C预定义符号,在预处理期间完成,可以直接使用。

有如下几个符号:

2.#define

在这个define情况下,该语句是在预处理期间处理完成的。有下面几种不同的分类:

2.1#define 常量定义

注:在使用这个常量定义时候,尽量不要在后面带上“;"这个分号(有需要除外)。

2.2#define 宏定义

#define机制包括了⼀个规定,允许把参数替换到⽂本中,这种实现通常称为宏(macro)或定义宏(definemacro)

注:

1.参数列表的左括号必须与name紧邻,如果两者之间有任何空⽩存在,参数列表就会被解释为stuff的⼀部分。

2.要带上小括号

3.不要使用带有改变参数本身性质的语法

宏在替换过程中,遵循以下规律:

  1. 在调⽤宏时,⾸先对参数进⾏检查,看看是否包含任何由#define定义的符号。如果是,它们⾸先被替换。
  2. 替换⽂本随后被插⼊到程序中原来⽂本的位置。对于宏,参数名被他们的值所替换。
  3. 最后,再次对结果⽂件进⾏扫描,看看它是否包含任何由#define定义的符号。如果是,就重复上述处理过程。

注:1.define可以出现其他define符号 2.在define中不能使用递归

2.3 宏与函数区别

3.4#undef 移除定义

3.5define与函数的命名约定

在通常情况下,宏一般全部大写命名,函数一般不全大写命名。

3.#与##操作符

#作用:将参数字符串化
##作用:参数字符串化并粘合其右边符号作为一个变量名称

3.1#的代码示例:

把字符串中的参数字符串化

#include<stdio.h>
#define my_printf(val,name) printf("the num of " #name " is %d",val);
int main()
{
  int a = 6;
  my_printf(a, a);
  return 0;
}

注意:1.简单的宏替换2.把替换的参数字符串化

3.2##的代码示例

造一个贴切返回类型的函数名

#define my_max(type) \
type type##_max(type x,type y)\
{\
return x>y?x:y;\
}\
my_max(int)
int main()
{
  int max = int_max(1, 2);
  printf("max = %d\n", max);
  return 0;
}

4.命名行定义

许多C的编译器提供了⼀种能⼒,允许在命令⾏中定义符号。⽤于启动编译过程。例如:当我们根据同⼀个源⽂件要编译出⼀个程序的不同版本的时候,这个特性有点⽤处。(假定某个程序中声明了⼀个某个⻓度的数组,如果机器内存有限,我们需要⼀个很⼩的数组,但是另外⼀个机器内存⼤些,我们需要⼀个数组能够⼤些。)

5.条件编译

在编译⼀个程序的时候我们如果要将⼀条语句(⼀组语句)编译或者放弃是很⽅便的。

条件编译用来处理调试性的代码,删除可惜,保留⼜碍事,所以我们可以选择性的编译或者是一些不同版本的代码。

5.1#if

这个跟if基本是一样的,如果if后面结果为真,那么下面这些代码(直到#end)就执行编译,为假就不参与编译。

下面是示例:

int main()
{
#if 1
  printf("hello world\n");
#endif
  return 0;
}

这里有个很坑的写法:

#include<stdio.h>
#include<windows.h>
#define A 1
int main()
{
  int a = 1;
#if A == a
  printf("hello world\n");
#endif
  system("pause");
  return 0;
}

明明相等为啥printf不参与编译啊?因为变量a是在预处理之后形成的。

5.2#ifdef

如果后面的参量名称被定义了,那么ifdef下面(直到#endif)就参与编译。如果没有被定义,ifdef下面(直到#endif)就不参与编译。

注:#ifdef一般逻辑比较简单,只与#else连用,不会和#elif进行连用,连用的话逻辑与我们想的也不太一样。

#include<stdio.h>
#include<windows.h>
#define PAUSE
int main()
{
#ifdef PAUSE
  printf("pause\n");
#endif
  return 0;
}

5.3#ifndef

如果后面的参量名称没有被定义,那么ifndef下面(直到#endif)就参与编译。如果被定义,ifndef下面(直到#endif)就不参与编译。

#include<stdio.h>
#include<windows.h>
#define PAUSE
int main()
{
#ifndef PAUSE
  printf("pause\n");
#endif
  return 0;
}

所以这里有个很重要的常见用法:就是避免头文件重复包含的问题(参见下面方法2)。

//1.方式1:
#pragma once
//2.方式2:
#ifndef ONCE
#define ONCE
//...
#endif

5.4多个分支条件编译#if #elif #else

与if、else if、else同理

#include<stdio.h>
#include<windows.h>
#define A 2
int main()
{
#if A==1
  printf("1\n");
#elif A==2
  printf("2\n");
#elif A==3
  printf("3\n");
#endif
  system("pause");
  return 0;
}

5.5上面所说的可以嵌套使用

#include<stdio.h>
#include<windows.h>
#define A 2
int main()
{
#if A==2
#ifdef A
  printf("ok\n");
#endif
#endif
  return 0;
}

6.头文件包含

#include”“先搜索本地头文件,找不到再去找库头文件。

#include<>直接去找库头文件.


完。

相关文章
|
10月前
|
编译器 C语言 C++
25 C语言 - 预处理器
25 C语言 - 预处理器
31 0
|
11月前
|
存储 编译器 程序员
【C语言】预处理
【C语言】预处理
43 0
|
3月前
|
编译器 C语言 C++
【C语言基础】:预处理详解(一)
【C语言基础】:预处理详解(一)
|
3月前
|
C语言 编译器 开发者
【C语言基础】:预处理详解(二)
【C语言基础】:预处理详解(二)
|
4月前
|
编译器 C语言
C语言:预处理
C语言:预处理
40 1
C语言:预处理
|
3月前
|
编译器 C语言
【C语言】:预处理详解
【C语言】:预处理详解
28 0
|
4月前
|
编译器 C语言 开发者
C语言的预处理
C语言的预处理
30 0
|
4月前
|
编译器 Linux C语言
C语言:预处理详解
C语言:预处理详解
|
4月前
|
编译器 C语言
C语言基础专题 - 预处理
C语言基础专题 - 预处理
21 1
|
4月前
|
C语言
【C语言进阶篇】你真的了解预处理吗? 预处理详细解析
【C语言进阶篇】你真的了解预处理吗? 预处理详细解析
52 0