网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
#define MAX 1000
#define reg register //为 register这个关键字,创建一个简短的名字
#define do\_forever for(;;) //用更形象的符号来替换一种实现
#define CASE break;case //在写case语句的时候自动把 break写上。
// 如果定义的 stuff过长,可以分成几行写,除了最后一行外,
//每行的后面都加一个反斜杠(续行符)。
#define DEBUG\_PRINT printf("file:%s\tline:%d\t \
date:%s\ttime:%s\n" ,\
\_\_FILE\_\_,\_\_LINE\_\_ , \
\_\_DATE\_\_,\_\_TIME\_\_ )
#define MAX 1000;
#define STP "hello"
#define print printf("hehe\n");
int main()
{
int m = MAX;
printf("%d\n", MAX);
printf("%s\n", STP);
print;
return 0;
}
✏️#define定义宏
#define 机制包括了一个规定,允许把参数替换到文本中,这种实现通常称为宏(macro)或定义 宏(define macro)。
宏的申明方式 :#define name( parament-list ) stuff其中的 parament-list 是一个由逗号隔开的符号表,它们可能出现在stuff中。
注意:
参数列表的左括号必须与name紧邻。
如果两者之间有任何空白存在,参数列表就会被解释为stuff的一部分。
#include <stdio.h>
#define SQUARE( x ) x \* x
int main()
{
int r = SQUARE(5);
//r = 5\*5;
//这里存在缺陷:
int a = SQUARE(5+1);
//认为是:a = (5+1)\*(5+1)?错误,由替换产生的表达式并没有按照预想的次序进行求值
//a = 5+1\*5+1 = 11
//在宏定义上加上两个括号,这个问题便轻松的解决了:
return 0;
}
#define SQUARE(x) ((x)\*(x))
int main()
{
int r = SQUARE(5);
int s = SQUARE(5 + 1);
printf("%d\n", r);
printf("%d\n", s);
return 0;
}
这里有一个问题:宏定义要不要把整体括号括起来?我们来看一段代码:
#include <stdio.h>
#define DOUBLE(x) (x)+(x)
int main()
{
int r = 10 \* DOUBLE(3);
printf("%d\n", r);
return 0;
}
结果为33,那如果整体有括号呢?
#include <stdio.h>
#define DOUBLE(x) ((x)+(x))
int main()
{
int r = 10 \* DOUBLE(3);
printf("%d\n", r);
return 0;
}
所以用于对数值表达式进行求值的宏定义都应该用这种方式加上括号,避免在使用宏时由于参数中的操作符或邻近操作符之间不可预料的相互作用。
✏️#define替换规则
在程序中扩展#define定义符号和宏时,需要涉及几个步骤:
- 在调用宏时,首先对参数进行检查,看看是否包含任何由#define定义的符号。如果是,它们首先 被替换。
- 替换文本随后被插入到程序中原来文本的位置。对于宏,参数名被他们的值所替换。
- 最后,再次对结果文件进行扫描,看看它是否包含任何由#define定义的符号。如果是,就重复上 述处理过程。
注意:
- 宏参数和#define 定义中可以出现其他#define定义的符号。但是对于宏,不能出现递归。
- 当预处理器搜索#define定义的符号的时候,字符串常量的内容并不被搜索。
✏️#和##两个预处理的工具
如何把参数插入到字符串中?
#include<stdio.h>
int main()
{
char\* p = "hello ""world\n";
printf("hello"" world\n");
printf("%s", p);
return 0;
}
字符串是有自动连接的特点的。
#define PRINT(FORMAT, VALUE)\
printf("the value is "FORMAT"\n", VALUE);
...
PRINT("%d", 10);
这里只有当字符串作为宏参数的时候才可以把字符串放在字符串中。
另外一个技巧是:
使用 # ,把一个宏参数变成对应的字符串。
#include<stdio.h>
#define PRINT( NAME, TYPE) printf("the value of "#NAME" is "TYPE"\n",NAME);
int main()
{
double a = 4.0;
PRINT(a, "%lf");
int b = 10;
PRINT(b, "%d");
return 0;
}
##可以把位于它两边的符号合成一个符号。 它允许宏定义从分离的文本片段创建标识符。
#include<stdio.h>
#define CAT(X,Y) X##Y
int main()
{
int class101 = 100;
printf("%d\n", CAT(class, 101));
printf("%d\n", class101);
return 0;
}
✏️带副作用的宏参数
当宏参数在宏的定义中出现超过一次的时候,如果参数带有副作用,那么你在使用这个宏的时候就可能 出现危险,导致不可预测的后果。副作用就是表达式求值的时候出现的永久性效果。
例如:
x+1;//不带副作用
x++;//带有副作用
✏️宏和函数的优缺点对比
宏有时候可以做函数做不到的事情。比如:宏的参数可以出现类型,但是函数做不到。
#define MALLOC(num, type) (type\*)malloc((num)\*sizeof(type))
#include<stdio.h>
int main()
{
//malloc(40);
//malloc(10, int);
int\*p = MALLOC(10, int);
//int\* p = (int\*)malloc((10) \* sizeof(int));
return 0;
}
✏️命名约定
一般来讲函数的宏的使用语法很相似;所以语言本身没法帮我们区分二者。
那我们平时的一个习惯是:
把宏名全部大写
函数名不要全部大写
🎉#undef
这条指令用于移除一个宏定义。
#undef NAME //如果现存的一个名字需要被重新定义,那么它的旧名字首先要被移除。
🎉命令行定义
许多C 的编译器提供了一种能力,允许在命令行中定义符号。用于启动编译过程。 例如:当我们根据同一个源文件要编译出不同的一个程序的不同版本的时候,这个特性有点用处。(假 定某个程序中声明了一个某个长度的数组,如果机器内存有限,我们需要一个很小的数组,但是另外一 个机器内存大写,我们需要一个数组能够大写。)
#include <stdio.h>
int main()
{
int array [ARRAY_SIZE];
int i = 0;
for(i = 0; i< ARRAY_SIZE; i ++)
{
array[i] = i;
}
for(i = 0; i< ARRAY_SIZE; i ++)
{
printf("%d " ,array[i]);
}
printf("\n" );
return 0;
}
编译指令:
//linux 环境
gcc -D ARRAY_SIZE=10 programe.c
🎉条件编译
常见的条件编译指令:
1.
#if 常量表达式
//...
#endif
//常量表达式由预处理器求值。
如:
#define \_\_DEBUG\_\_ 1
#if \_\_DEBUG\_\_
//..
#endif
2.多个分支的条件编译
#if 常量表达式
//...
#elif 常量表达式
//...
#else
//...
#endif
3.判断是否被定义
#if defined(symbol)
#ifdef symbol
#if !defined(symbol)
#ifndef symbol
4.嵌套指令
#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
#endif
🎉文件包含
#include 指令可以使另外一个文件被编译;就像它实际出现于 #include 指令的地方一样。
这种替换的方式很简单:
预处理器先删除这条指令,并用包含文件的内容替换。
这样一个源文件被包含10次,那就实际被编译10次。
✏️头文件包含
本地文件包含
#include "filename"
库文件包含
#include <filename.h>
使用<>和“ ”文件包含的区别
<> :直接去库目录(标准路径)下查找, 如果该头文件未找到 ,提示编译错误。
" " :先去代码所在的路径下查找,如果该头文件未找到,再去库目录(标准路径)下查找,如果找不到,提示编译错误。
✏️嵌套文件包含
如果出现这样的场景
comm.h和comm.c是公共模块。
test1.h和test1.c使用了公共模块。
test2.h和test2.c使用了公共模块。
test.h和test.c使用了test1模块和test2模块。
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
如果出现这样的场景
comm.h和comm.c是公共模块。
test1.h和test1.c使用了公共模块。
test2.h和test2.c使用了公共模块。
test.h和test.c使用了test1模块和test2模块。
[外链图片转存中…(img-L70s3soq-1715549949270)]
[外链图片转存中…(img-Uob76kti-1715549949271)]
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!