Linux环境下C语言常用宏定义

http://baike.baidu.com/view/2714887.htm


l  防止一个头文件被重复包含

          #ifndefCOMDEF_H

   #define COMDEF_H

   //头文件内容

   #endif

 

l  重新定义一些类型,防止由于各种平台和编译器的不同,而产生的类型字节数差异,方便移植。

         typedefunsigned char boolean; /* Boolean value type. */

   typedef unsigned longint uint32; /* Unsigned 32 bit value */

   typedef unsigned short  uint16; /* Unsigned 16 bit value */

   typedef unsigned char  uint8; /* Unsigned 8 bit value */

   typedef signed longint int32; /* Signed 32 bit value */

   typedef signed short int16; /* Signed 16 bit value */

   typedef signed char int8;/* Signed 8 bit value */

  

//下面的不建议使用

  typedef unsigned char byte; /* Unsigned8 bit value type. */

  typedef unsigned short word; /* Unsinged16 bit value type. */

  typedef unsigned long dword; /* Unsigned32 bit value type. */

  typedef unsigned char uint1; /* Unsigned8 bit value type. */

  typedef unsigned short uint2; /*Unsigned 16 bit value type. */

  typedef unsigned long uint4; /* Unsigned32 bit value type. */

  typedef signed char int1; /* Signed 8bit value type. */

  typedef signed short int2; /* Signed 16bit value type. */

  typedef long int int4; /* Signed 32 bitvalue type. */

  typedef signed long sint31; /* Signed 32bit value */

  typedef signed short sint15; /* Signed16 bit value */

typedef signed char sint7; /* Signed 8bit value */

 

l  得到指定地址上的一个字节或字

   #define MEM_B(x)(*((byte *)(x)))

   #define MEM_W(x)(*((word *)(x)))

 

l  求最大值和最小值

   #define MAX(x,y)(((x)>(y)) ? (x) : (y))

   #define MIN(x,y) (((x)< (y)) ? (x) : (y))

 

l  得到一个field在结构体(struct)中的偏移量

   #defineFPOS(type,field) ((dword)&((type *)0)->field)

 

l  得到一个结构体中field所占用的字节数

   #defineFSIZ(type,field) sizeof(((type *)0)->field)

 

l  按照LSB格式把两个字节转化为一个Word

   #define FLIPW(ray)((((word)(ray)[0]) * 256) + (ray)[1])

 

l  按照LSB格式把一个Word转化为两个字节

   #define FLOPW(ray,val)(ray)[0] = ((val)/256); (ray)[1] = ((val) & 0xFF)

 

l  得到一个变量的地址(word宽度)

   #define B_PTR(var)((byte *) (void *) &(var))

   #define W_PTR(var)((word *) (void *) &(var))

 

l  得到一个字的高位和低位字节

   #define WORD_LO(xxx)((byte) ((word)(xxx) & 255))

   #define WORD_HI(xxx)((byte) ((word)(xxx) >> 8))

 

l  返回一个比X大的最接近的8的倍数

   #define RND8(x) ((((x)+ 7)/8) * 8)

 

l  将一个字母转换为大写

   #define UPCASE(c)(((c)>='a' && (c) <= 'z') ? ((c) - 0x20) : (c))

 

l  判断字符是不是10进值的数字

   #define DECCHK(c)((c)>='0' && (c)<='9')

 

l  判断字符是不是16进值的数字

   #define HEXCHK(c)(((c) >= '0' && (c)<='9') ((c)>='A' && (c)<= 'F') \

   ((c)>='a'&& (c)<='f'))

 

l  防止溢出的一个方法

   #define INC_SAT(val)(val=((val)+1>(val)) ? (val)+1 : (val))

 

l  返回数组元素的个数

   #define ARR_SIZE(a)(sizeof((a))/sizeof((a[0])))

 

l  返回一个无符号数n尾的值MOD_BY_POWER_OF_TWO(X,n)=X%(2^n)

   #defineMOD_BY_POWER_OF_TWO( val, mod_by ) ((dword)(val) & (dword)((mod_by)-1))

 

l  对于IO空间映射在存储空间的结构,输入输出处理

   #define inp(port)(*((volatile byte *)(port)))

   #define inpw(port)(*((volatile word *)(port)))

   #define inpdw(port)(*((volatile dword *)(port)))

   #define outp(port,val)(*((volatile byte *)(port))=((byte)(val)))

   #define outpw(port,val) (*((volatile word *)(port))=((word)(val)))

   #define outpdw(port,val) (*((volatile dword *)(port))=((dword)(val)))

l  使用一些宏跟踪调试

   ANSI标准说明了五个预定义的宏名。它们是:

   __LINE__

   __FILE__

   __DATE__

   __TIME__

   __STDC__

   C++中还定义了 __cplusplus

   如果编译器不是标准的,则可能仅支持以上宏名中的几个,或根本不支持。记住编译程序也许还提供其它预定义的宏名。

  __LINE__ 及__FILE__ 宏指示,#line指令可以改变它的值,简单的讲,编译时,它们包含程序的当前行数和文件名。

  __DATE__ 宏指令含有形式为月/日/年的串,表示源文件被翻译到代码时的日期。

  __TIME__ 宏指令包含程序编译的时间。时间用字符串表示,其形式为: 分:秒

  __STDC__ 宏指令的意义是编译时定义的。一般来讲,如果__STDC__已经定义,编译器将仅接受不包含任何非标准扩展的标准C/C++代码。如果实现是标准的,则宏__STDC__含有十进制常量1。如果它含有任何其它数,则实现是非标准的。

  __cplusplus 与标准c++一致的编译器把它定义为一个包含至少6为的数值。与标准c++不一致的编译器将使用具有5位或更少的数值。

  可以定义宏,例如:

  当定义了_DEBUG,输出数据信息和所在文件所在行

  #ifdef _DEBUG

  #define DEBUGMSG(msg,date)printf(msg);printf(“%d%d%d”,date,_LINE_,_FILE_)

  #else

  #define DEBUGMSG(msg,date)

#endif

 

l  宏定义防止错误使用小括号包含。

  例如:

  有问题的定义:#define DUMP_WRITE(addr,nr){memcpy(bufp,addr,nr); bufp += nr;}

  应该使用的定义: #difne DO(a,b) do{a+b;a++;}while(0)

  例如:

  if(addr)

  DUMP_WRITE(addr,nr);

  else

  do_somethong_else();

  宏展开以后变成这样:

  if(addr)

  {memcpy(bufp,addr,nr); bufp += nr;};

  else

  do_something_else();

  gcc在碰到else前面的“;”时就认为if语句已经结束,因而后面的else不在if语句中。而采用do{}while(0)的定义,在任何情况下都没有问题。而改为#difne DO(a,b) do{a+b;a++;}while(0) 的定义则在任何情况下都不会出错。


  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值