• 宏相关使用


    一、简述

    1.1宏

    #define命令是C语言中的一个宏定义命令,它用来将一个标识符定义为一个字符串,该标识符被称为宏名,被定义的字符串称为替换文本

    1.2宏原理

    • C源码到可执行程序过程实际经过:预处理、编译、汇编和连接几个过程。
    • 其中预处理器产生编译器的输出,它实现以下的功能(宏展开在预处理阶段展开):
      • (1)文件包含
        • 可以把源程序中的#include 扩展为文件正文,即把包含的.h文件找到并展开到#include 所在处。
      • (2)条件编译
        • 预处理器根据#if和#ifdef等编译命令及其后的条件,将源程序中的某部分包含进来或排除在外,通常把排除在外的语句转换成空行。
      • (3)宏展开
        • 预处理器将源程序文件中出现的对宏的引用展开成相应的宏 定义,即本文所说的#define的功能,由预处理器来完成。

    经过预处理器处理的源程序与之前的源程序有所有不同,在这个阶段所进行的工作只是纯粹的替换与展开,没有任何计算功能,

    二、宏使用

    2.1宏分类

    一般可以分为两类:

    1. 不带参数的宏;
    2. 带指定个数参数的宏定义;
    3. 带可变个数参数。

    2.1.1不带参数宏

    1. 简单的宏定义:

    #define <宏名> <字符串>

    例: #define PI 3.1415926

    2.1.2带指定个数参数的宏定义

    #define <宏名> (<参数表>) <宏体>

    例: #define A(x) x

    2.1.3带可变个数参数

    1999 年的 ISO C 标准中,宏可以声明为接受可变数量的参数,就像函数一样。定义宏的语法类似于函数的语法。

    例子:

    #define debug(format, ...) fprintf (stderr, format, __VA_ARGS__)

    说明:这里的是一个可变参数。在调用这样的宏时,它代表零个或多个参数。这组标记替换了宏主体中出现的标识符 __VA_ARGS__。

    GCC 长期以来一直支持可变参数宏,并使用不同的语法,允许您像任何其他参数一样为变量参数命名。

    下面是一个例子:

    #define debug(format, args...) fprintf (stderr, format, args)

    这在所有方面都等同于上面的 ISO C 示例,但可以说更具可读性和描述性。

    但是使用过上面两种形式的可变参数宏之后会发现一个问题,就是不能不传变量参数。

    在标准 C 中,不允许完全忽略变量参数;但是你可以传递一个空参数。例如,调用在 ISO C 中无效,因为字符串后没有逗号:

    debug ("A message")

    为了解决这个问题,CPP 对与标记粘贴运算符“##”一起使用的变量参数进行了特殊处理。形式如下:

    #define debug(format, ...) fprintf (stderr, format, ## __VA_ARGS__)

    如果变量参数被省略或为空,“##”运算符会导致预处理器删除它前面的逗号。如果在宏调用中提供了一些变量参数,GNU CPP 会将变量参数放在逗号之后。就像任何其他粘贴的宏参数一样,这些参数不是宏扩展的。

    三、宏替换注意事项


    3.1 简单宏中问题

    宏展开只是字符替换,不会考虑替换相关字符以后优先级,替换过程也不会进行相关计算。

    示例:

    在简单宏定义的使用中,当替换文本所表示的字符串为一个表达式时,容易引起误解和误用。如下例:

      1 #define N 2+2
      2 
      3 void main()
      4 
      5 {
      6 
      7 int a=N*N;
      8 
      9 printf(“%d”,a);
     10 
     11 }
     12 


    (1) 出现问题

    在此程序中存在着宏定义命令,宏N代表的字符串是2+2,在程序中有对宏N的使用,一般同学在读该程序时,容易产生的问题是先求解N为2+2=4,然后在程序中计算a时使用乘法,即N*N=4*4=16,其实该题的结果为8,为什么结果有这么大的偏差?

    (2) 问题解析

    如1节所述,宏展开是在预处理阶段完成的,这个阶段把替换文本只是看作一个字符串,并不会有任何的计算发生,在展开时是在宏N出现的地方 只是简单地使用串2+2来代替N,并不会增添任何的符号,所以对该程序展开后的结果是a=2+2*2+2,计算后=8,这就是宏替换的实质,如何写程序才能完成结果为16的运算呢?

    (3)解决办法

    /*将宏定义写成如下形式*/

    #define N (2+2)

    /*这样就可替换成(2+2)*(2+2)=16*/

    3.2 带参数宏问题

    在带参数的宏定义的使用中,极易引起误解。例如我们需要做个宏替换能求任何数的平方,这就需要使用参数,以便在程序中用实际参数来替换宏定义中的参数。一般学生容易写成如下形式:

      1 #define area(x) x*x
      2 
      3 /*这在使用中是很容易出现问题的,看如下的程序*/
      4 
      5 void main()
      6 
      7 {
      8 
      9 int y = area(2+2);
     10 
     11 printf(“%d”,y);
     12 
     13 }
     14 

    按理说给的参数是2+2,所得的结果应该为4*4=16,但是错了,因为该程序的实际结果为8,仍然是没能遵循纯粹的简单替换的规则,又是先计算再替换 了,在这道程序里,2+2即为area宏中的参数,应该由它来替换宏定义中的x,即替换成2+2*2+2=8了。那如果遵循(1)中的解决办法,把2+2 括起来,即把宏体中的x括起来,是否可以呢?#define area(x) (x)*(x),对于area(2+2),替换为(2+2)*(2+2)=16,可以解决,但是对于area(2+2)/area(2+2)又会怎么样呢,有的学生一看到这道题马上给出结果,因为分子分母一样,又错了,还是忘了遵循先替换再计算的规则了,这道题替换后会变为 (2+2)*(2+2)/(2+2)*(2+2)即4*4/4*4按照乘除运算规则,结果为16/4*4=4*4=16,那应该怎么呢?解决方法是在整个宏体上再加一个括号,即#define area(x) ((x)*(x)),不要觉得这没必要,没有它,是不行的。

    要想能够真正使用好宏定义,那么在读别人的程序时,一定要记住先将程序中对宏的使用全部替换成它所代表的字符串,不要自作主张地添加任何其他符号,完全展开后再进行相应的计算,就不会写错运行结果。

    如果是自己编程使用宏替换,则在使用简单宏定义时,当字符串中不只一个符号时,加上括号表现出优先级,如果是带参数的宏定义,则要给宏体中的每个参数加上括号,并在整个宏体上再加一个括号。看到这里,不禁要问,用宏定义这么麻烦,这么容易出错,可不可以摒弃它, 那让我们来看一下在C语言中用宏定义的好处吧。

    如:

      1 #include <iostream.h>
      2 
      3 #define product(x) x*x
      4 
      5 int main()
      6 
      7 {
      8 
      9 int i=3;
     10 
     11 int j,k;
     12 
     13 j = product(i++);
     14 
     15 cout<<"j="<<j<<endl;
     16 
     17 cout<<"i="<<i<<endl;
     18 
     19 k = product(++i);
     20 
     21 cout<<"k="<<k<<endl;
     22 
     23 cout<<"i="<<i<<endl;
     24 
     25 return 0;
     26 
     27 }
     28 

    依次输出结果:

    j=9;i=5;k=49;i=7

    四、宏define中的三个特殊符号:#,##,#@

    ##连接

    #@加单引号

    #加双引号

    示例:

      1 #define Conn(x,y) x##y
      2 
      3 #define ToChar(x) #@x
      4 
      5 #define ToString(x) #x
      6 
    • x##y表示什么?表示x连接y

    举例说:

      1 int n = Conn(123,456); /* 结果就是n=123456;*/
      2 
      3 char* str = Conn("asdf", "adf"); /*结果就是 str = "asdfadf";*/
      4 
      5 #define conn(a,b) a##e##b
      6 

    e:是冥

    int c = conn(10,2);

    结果c=100

    • 再来看#@x,其实就是给x加上单引号,结果返回是一个const char。

    举例说:

    char a = ToChar(1);结果就是a='1';

    做个越界试验char a = ToChar(123);结果就错了;

    但是如果你的参数超过四个字符,编译器就给给你报错了!

    error C2015: too many characters in constant :P

    • 最后看看#x,估计你也明白了,他是给x加双引号

    char* str = ToString(123132);就成了str="123132";

    五、常用宏

    • 防止一个头文件被重复包含
      1 #ifndef BODYDEF_H
      2 
      3 #define BODYDEF_H
      4 
      5 //头文件内容
      6 
      7 #endif
      8 
    • 得到指定地址上的一个字节或字
      1 #define MEM_B( x ) ( *( (byte *) (x) ) )
      2 
      3 #define MEM_W( x ) ( *( (word *) (x) ) )
      4 
      5 用法如下:
      6 
      7 #include <iostream>
      8 
      9 #include <windows.h>
     10 
     11 #define MEM_B(x) (*((byte*)(x)))
     12 
     13 #define MEM_W(x) (*((WORD*)(x)))
     14 
     15 int main()
     16 
     17 {
     18 
     19 int bTest = 0x123456;
     20 
     21 byte m = MEM_B((&bTest));/*m=0x56*/
     22 
     23 int n = MEM_W((&bTest));/*n=0x3456*/
     24 
     25 return 0;
     26 
     27 }
     28 
    • 得到一个field在结构体(struct)中的偏移量
      1 #define OFFSETOF( type, field ) ( (size_t) &(( type *) 0)-> field )
      2 
      3 请参考文章:详解写宏定义:得到一个field在结构体(struct type)中的偏移量。
      4 
    • 得到一个结构体中field所占用的字节数
      1 #define FSIZ( type, field ) sizeof( ((type *) 0)->field )
    • 得到一个变量的地址(word宽度)
      1 #define B_PTR( var ) ( (byte *) (void *) &(var) )
      2 
      3 #define W_PTR( var ) ( (word *) (void *) &(var) )
      4 
    • 将一个字母转换为大写
      1 #define UPCASE( c ) ( ((c) >= ''a'' && (c) <= ''z'') ? ((c) - 0x20) : (c) )
    • 判断字符是不是10进值的数字
      1 #define DECCHK( c ) ((c) >= ''0'' && (c) <= ''9'')
    • 判断字符是不是16进值的数字
      1 #define HEXCHK( c ) ( ((c) >= ''0'' && (c) <= ''9'') ||((c) >= ''A'' && (c) <= ''F'') ||((c) >= ''a'' && (c) <= ''f'') )
    • 防止溢出的一个方法
      1 #define INC_SAT( val ) (val = ((val)+1 > (val)) ? (val)+1 : (val))
    • 返回数组元素的个数
      1 #define ARR_SIZE( a ) ( sizeof( (a) ) / sizeof( (a[0]) ) )
    • 使用一些宏跟踪调试

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

    _LINE_ /*(两个下划线),对应%d*/

    _FILE_ /*对应%s*/

    _DATE_ /*对应%s*/

    _TIME_ /*对应%s*/

    __FILE__:作用:表示当前源文件名,类型为字符串常量;

    __LINE__:作用:代表当前程序行的行号,类型为十进制整数常量;

    #line:语法:#line 行号 [“文件名”]

    作用:将行号和文件名更改为指定的行号和文件名;

    __func__ 和 __FUNCTION__

    作用:代表当前函数的函数名,类型为字符串常量;

    __DATE__:作用:代表日期,形式为Mmm dd yyyy 的字符串常量;

    __TIME__:作用:代表时间,hh:mm:ss 形式的字符串型常量;

  • 相关阅读:
    Spring EL Operators example
    Spring EL method invocation example
    Spring EL hello world example
    Spring @PostConstruct and @PreDestroy example
    Spring init-method and destroy-method example
    Define custom @Required-style annotation in Spring
    Spring dependency checking with @Required Annotation
    Spring properties dependency checking
    UESTC 883 方老师与两个串 --二分搜索+DP
    UESTC 882 冬马党 --状压DP
  • 原文地址:https://www.cnblogs.com/mehome/p/16110474.html
Copyright © 2020-2023  润新知