首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 开发语言 > C++ >

宏中的#如何表示

2012-05-16 
宏中的#怎么表示我想写一对宏,在MFC中检测内存泄漏,如下:C/C++ code#define BEGIN_MEMTEST#ifdef _DEBUG \

宏中的#怎么表示
我想写一对宏,在MFC中检测内存泄漏,如下:

C/C++ code
#define BEGIN_MEMTEST    #ifdef _DEBUG \    CMemoryState oldMemState, newMemState, diffMemState; \    oldMemState.Checkpoint(); \#endif#define END_MEMTEST        #ifdef _DEBUG \                                    newMemState.Checkpoint(); \                                    if(diffMemState.Difference(oldMemState, newMemState)) {  \                                    TRACE(_T(FunName)); \                                    diffMemState.DumpStatistics(); \                                    } \                                    #endif 


编译提示如下:
G:\memtest\memtestView.cpp(60) : error C2121: '#' : invalid character : possibly the result of a macro expansion
G:\memtest\memtestView.cpp(60) : error C2065: 'ifdef' : undeclared identifier
G:\memtest\memtestView.cpp(60) : error C2143: syntax error : missing ';' before 'constant'
G:\memtest\memtestView.cpp(60) : error C2146: syntax error : missing ';' before identifier 'CMemoryState'
G:\memtest\memtestView.cpp(60) : error C2146: syntax error : missing ';' before identifier 'oldMemState'
G:\memtest\memtestView.cpp(60) : error C2275: 'CMemoryState' : illegal use of this type as an expression
  c:\program files\microsoft visual studio\vc98\mfc\include\afx.h(1688) : see declaration of 'CMemoryState'
G:\memtest\memtestView.cpp(60) : error C2065: 'oldMemState' : undeclared identifier
G:\memtest\memtestView.cpp(60) : error C2065: 'newMemState' : undeclared identifier
G:\memtest\memtestView.cpp(60) : error C2065: 'diffMemState' : undeclared identifier
G:\memtest\memtestView.cpp(60) : error C2228: left of '.Checkpoint' must have class/struct/union type
G:\memtest\memtestView.cpp(60) : error C2121: '#' : invalid character : possibly the result of a macro expansion
G:\memtest\memtestView.cpp(61) : error C2065: 'endif' : undeclared identifier
G:\memtest\memtestView.cpp(61) : error C2146: syntax error : missing ';' before identifier 'pDC'
G:\memtest\memtestView.cpp(65) : error C2121: '#' : invalid character : possibly the result of a macro expansion
G:\memtest\memtestView.cpp(65) : error C2143: syntax error : missing ';' before 'constant'
G:\memtest\memtestView.cpp(65) : error C2146: syntax error : missing ';' before identifier 'newMemState'
G:\memtest\memtestView.cpp(65) : error C2228: left of '.Checkpoint' must have class/struct/union type
G:\memtest\memtestView.cpp(65) : error C2228: left of '.Difference' must have class/struct/union type
G:\memtest\memtestView.cpp(65) : error C2065: 'FunName' : undeclared identifier
G:\memtest\memtestView.cpp(65) : error C2228: left of '.DumpStatistics' must have class/struct/union type
G:\memtest\memtestView.cpp(65) : error C2121: '#' : invalid character : possibly the result of a macro expansion
G:\memtest\memtestView.cpp(68) : error C2143: syntax error : missing ';' before '}'
Generating Code...


我知道在宏中#可以转化字符串,##连接字符串,可是我要在宏中使用正常的#号怎么写呢?比如 #define BEGIN_MEMTEST#ifdef _DEBUG

[解决办法]
C/C++ code
#ifdef _DEBUG#define BEGIN_MEMTEST     \    CMemoryState oldMemState, newMemState, diffMemState; \    oldMemState.Checkpoint(); \#else#define BEGIN_MEMTEST#endif
[解决办法]
探讨

C/C++ code
#ifdef _DEBUG
#define BEGIN_MEMTEST \
CMemoryState oldMemState, newMemState, diffMemState; \
oldMemState.Checkpoint(); \
#else
#define BEGIN_MEMTEST
#endif

[解决办法]
探讨
C/C++ code


#ifdef _DEBUG
#define BEGIN_MEMTEST \
CMemoryState oldMemState, newMemState, diffMemState; \
oldMemState.Checkpoint(); \
#else
#define BEGIN_MEMTEST
#endif


[解决办法]
#与##在宏定义中的--宏展开
#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

[解决办法]
不支持嵌套,而且可读性也差。

热点排行