读书人

以上的宏怎么理解

发布时间: 2012-09-17 12:06:51 作者: rapoo

以下的宏如何理解?
这是linux里的hook定义,我觉得和函数指针的定义似乎有些不同。。。。。
请各位赐教。多谢!
本人学艺不精,烦请说详细一点。。。。。。

#define NF_HOOK(pf, hook, skb, indev, outdev, okfn) \
NF_HOOK_THRESH(pf, hook, skb, indev, outdev, okfn, INT_MIN)


#define NF_HOOK_THRESH(pf, hook, skb, indev, outdev, okfn, thresh) \
({int __ret; \
if ((__ret=nf_hook_thresh(pf, hook, (skb), indev, outdev, okfn, thresh, 1)) == 1)\
__ret = (okfn)(skb); \
__ret;})

[解决办法]
1.NF_HOOK函数直接宏替换为另外一个函数NF_HOOK_THRESH,且后者多了一个参数INT_MIN
2.宏定义NF_HOOK_THRESH函数
这里不是定义函数指针了,是宏定义函数了
[解决办法]
尝试着把一个宏用另一个代替,就能知道什么意思了。
[解决办法]
宏不是函数,它是根据一系列预定义的规则替换一定的文本模式。预编译器会直接将宏展开,之后再参与编译。

探讨

#define NF_HOOK(pf, hook, skb, indev, outdev, okfn) \
NF_HOOK_THRESH(pf, hook, skb, indev, outdev, okfn, INT_MIN)

哦,把NF_HOOK看成函数定义,除了没有返回值,其它的倒是可以理解了。

可是,这种定义方法,出处哪里?没有返回值行吗?那就是void函数?

[解决办法]
宏定义就是简单的替换。。。。记着这点就OK了
[解决办法]
探讨
宏就是简单的文本替换,自己换下就知道什么意思了

[解决办法]
宏定义就是简单替换。有优缺点:

1)优点,宏定义可以再用的地方直接展开,优点类似内联函数,省去了重复编码的工作;
2)缺点,宏定义并不是函数的定义,会出现意想不到副作用,比如:在宏定义的函数里,函数参数类型不做检查,阅读代码费解……;宏定义的函数参数如果带有表达式容易出错。

建议初学者谨慎使用。
[解决办法]
#与##在宏定义中的--宏展开
#include <stdio.h>
#define f(a,b) a##b
#define g(a) #a
#define h(a) g(a)
int main()
{
printf("%s\n", h(f(1,2))); // 12
printf("%s\n", g(f(1,2))); // f(1,2)
return 0;
}
宏展开时:
如果宏定义以#开头,不展开参数,直接替换。
故g(f(1,2))--->#f(1,2)--->"f(1,2)";
如果宏定义不以#开头,展开参数,直接替换,由外层向里层,如果碰到的是#开头的宏,不继续往里层展开,往外层展开。
由外层向里层,如果碰到的是以非#开头的宏,继续往里层走,直至最里层,开始一层层往外层展开。
故h(f(1,2))--->h(12)--->g(12)---->#12----->"12"。
PS:
##在宏中定义,是字符连接符
如a##b##c 等同于 "abc"
#在宏开头出现,是表示宏展开的方式不同
#a 等同于"a"
#abc 等同于 "abc"
复杂的:
#include <stdio.h>
#define f(a,b) a##b
#define g(a) #a
#define h(a) g(a)
int main()
{
char a = 'a';
cout<<g(a)<<endl; // a
cout<<g(g(a))<<endl; // a
printf("%s\n", h(f(1,2))); // 12
printf("%s\n", g(f(1,2))); // f(1,2)
printf("%s\n", g(h(f(1,2)))); // h(f(1,2))
printf("%s\n", h(g(f(1,2)))); // "f(1,2)"
printf("%s\n", h(h(f(1,2)))); // "12"
system("pause");
return 0;
}
预处理后的:(在编译选项中添加/EP /P后编译生成的.i文件)
int main()
{
char a = 'a';
cout<<"a"<<endl;
cout<<"g(a)"<<endl;
printf("%s\n", "12");
printf("%s\n", "f(1,2)");
printf("%s\n", "h(f(1,2))");
printf("%s\n", "\"f(1,2)\"");
printf("%s\n", "\"12\"");
system("pause");
return 0;
}
---------------------------------------------------
宏解析
1. ##操作符
##操作符它的作用是在替代表中将其前后的参数连接成为一个预处理符号,它不能出现于宏替代表的开端和末尾。
例:
#define concat(s,t) s##t


#define AAA ABC
concat(A, AA)
将被替换成
ABC
2. 重新扫描和替换
在替换列表中的所有参数替换过之后,预处理器将对结果token序列重新扫描以便对其中的宏再次替换。
当正在替换的宏在其替换列表中发现自身时,就不再对其进行替换。今儿,在任何正在嵌套替换的宏的替换过程中遇到正被替换的宏就对其不再进行替换(防止递归)。
例:
#define ROOT AAA CCC
#define AAA ROOT
ROOT
将被替换成
ROOT CCC

综上所述,宏定义不是简单替换。

读书人网 >C语言

热点推荐