一、#define
1、仅仅作为标识符:
#define作为标识符时是没有参数的
2、作为宏:
#define 作为宏与它作为标识符的区别就是宏有参数而标识符没有
例如:#define MAX(X,Y) ((X)>(Y)?(X):(Y))
二、宏与函数的区别
1、宏在使用时(编译阶段)会直接被替换掉(即简单的代码复制),并且不会进行逻辑检测
2、在对代码调试时,不能进入内部调试,执行后就直接回产生值。而函数与此不同
3、宏在进行定义时不会考虑参数的类型(但这不安全)
4、宏会产生带副作用的宏参数
请分析下面这段代码:
#include<stdio.h> #include<stdlib.h> #define MAX(X,Y) ((X)>(Y)?(X):(Y)) int main() { int a = 10; int b = 20; int ret = MAX(a++,b++);//int ret = MAX(a++,b++) ((a++)>(b++)?(a++):(b++)) printf("a=%d,b=%d,ret=%d\n",a,b,ret); system("pause"); return 0; }
其产生的结果为:a=11,b=22,ret=21
产生这种结果的原因就是宏被替换后产生了带副作用的参数,a被执行了一次,b被执行了两次
5、参数宏的运行速度比函数快,因为它只是简单的替换,不需要参数压栈/出栈操作
6、宏在定义时要尽量多加括号
例如:在宏定义中 #define DOUBLE(X,Y) X*Y ,如果使用时给的参数是
(5+5,5+5),运行的结果就会出错。
7、宏可以将类型作为参数,而函数却不可以 (关于动态内存开辟)
用函数动态开辟一块空间:
int main() { int *p=(int *)malloc(10*sizeof(int)); int i=0; for(; i<10; i++) { p[i]=i; } for(i=0; i<10; i++) { printf("%d ",p[i]); } free(p); system("pause"); return 0; }
缺点:有可能会开辟内存失败。例如:int *p=(int *)malloc(0xffffffff);
0xffffffff 是所有内存地址编码的集合,即为所有内存,这种情况下,开辟内存就会失败,所 以在用 malloc()函数后还应该对指针 p 进行判断
int i=0; int *p=(int *)malloc(10*sizeof(int)); if(p == NULL) { printf("put of memory!\n"); exit(1); //结束程序 }
而利用宏来开辟内存就会省掉这些麻烦:
#define MALLOC(COUNT,TYPE) (TYPE*)alloc(COUNT*sizeof(TYPE)) void *alloc(int sz) { void *p=malloc(sz); if(p == NULL) { printf("out of memory!\n"); } return p; } int main() { int *p=MALLOC(10,int); int i=0; for(; i<10; i++) { p[i]=i; } for(i=0; i<10; i++) { printf("%d ",p[i]); } free(p); //动态开辟后一定要释放这些内存,否则会发生内存泄漏 system("pause"); return 0; }
开辟内存时只要传进类型就ok了
8、宏不可以递归,函数可以
#define MAX(X,Y) ((X)>(Y)?(X):(Y)) #define M 100 int main() { int ret = MAX(MAX(M,20),200); //这是嵌套使用,不是递归 return 0; }
9、函数只在目标文件中存在一处,比较节省空间
10、宏进行定义时,名字与参数之间不能有空格分开,但使用时可以有
时间: 2024-10-10 22:44:07