一、#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、宏进行定义时,名字与参数之间不能有空格分开,但使用时可以有
转载于:https://blog.51cto.com/10969583/1760114