Chinaunix首页 | 论坛 | 博客
  • 博客访问: 218183
  • 博文数量: 72
  • 博客积分: 3890
  • 博客等级: 中校
  • 技术积分: 810
  • 用 户 组: 普通用户
  • 注册时间: 2009-01-05 20:00
文章分类

全部博文(72)

文章存档

2010年(20)

2009年(52)

我的朋友

分类: LINUX

2009-02-15 23:13:57

  ##连接符号由两个井号组成,其功能是在带参数的宏定义中将两个子串(token)联接起来,从而形成一个新的子串。但它不可以是第一个或者最后一个子 串。所谓的子串(token)就是指编译器能够识别的最小语法单元。具体的定义在编译原理里有详尽的解释,但不知道也无所谓。同时值得注意的是#符是把传 递过来的参数当成字符串进行替代。下面来看看它们是怎样工作的。这是MSDN上的一个例子。 
   
  假设程序中已经定义了这样一个带参数的宏: 
   
  #define  paster(  n  )  printf(  "token"  #n  "  =  %d",  token##n  ) 
  同时又定义了一个整形变量: 
   
  int  token9  =  9; 
  现在在主程序中以下面的方式调用这个宏: 
   
  paster(  9  ); 
  那么在编译时,上面的这句话被扩展为: 
   
  printf(  "token"  "9"  "  =  %d",  token9  ); 
  注意到在这个例子中,paster(9);中的这个”9”被原封不动的当成了一个字符串,与”token”连接在了一起,从而成为了token9。而#n也被”9”所替代。 
   
  可想而知,上面程序运行的结果就是在屏幕上打印出token9=9 
   
  在ATL的编程中,我们查看它的源代码就会经常看见这样的一段: 
   
  #define  IMPLEMENTS_INTERFACE(Itf)  \ 
  {&IID_##Itf,  ENTRY_IS_OFFSET,BASE_OFFSET(_ITCls,  Itf)  }, 
  我们经常不假思索的这样使用它: 
  …… 
  IMPLEMENTS_INTERFACE(ICat) 
  …… 
   
  实际上IID_ICat  已经在别的地方由ATL向导定义了。当没有向导的时候,你只要遵循把IID_加在你的接口名前面来定义GUID的规则就也可以使用这个宏。在实际的开发过 程中可能很少用到这种技巧,但是ATL使用得如此广泛,而其中又出现了不少这样的源代码,所以明白它是怎么一回事也是相当重要的。我的一个朋友就是因为不 知道IMPLEMENTS_INTERFACE宏是怎么定义的,而又不小心改动了IID_ICat的定义而忙活了一整天。 
阅读(860) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~