宏定义用法的全面探究

作者: Peak_One | 来源:发表于2016-03-22 23:47 被阅读673次
    废话

    由于昨天一直在学习RunLoop,发现RunLoop不是我想象的那么简单,结果昨天一直在搞RunLoop,所以没有发博客,所以,今天一早必须得把昨天的博客给补出来,于是,在这里给大家分享一下宏定义的使用。希望对大家有帮助。

    概述

    众所周知,宏定义属于属于预编译指令,即在编译之前就已经将宏定义加载到了内存中。

    常用的宏定义指令
    #define 定义一个预处理宏
    #undef 取消宏的定义
    #include 包含文件命令
    #include_next 与#include相似, 但它有着特殊的用途
    #if 编译预处理中的条件命令, 相当于C语法中的if语句
    #ifdef 判断某个宏是否被定义, 若已定义, 执行随后的语句
    #ifndef 与#ifdef相反, 判断某个宏是否未被定义
    #elif 若#if, #ifdef, #ifndef或前面的#elif条件不满足, 则执行#elif之后的语句, 相当于C语法中的else-if
    #else 与#if, #ifdef, #ifndef对应, 若这些条件不满足, 则执行#else之后的语句, 相当于C语法中的else
    #endif #if, #ifdef, #ifndef这些条件命令的结束标志.
    #defined 与#if, #elif配合使用, 判断某个宏是否被定义
    #line 标志该语句所在的行号
    # 将宏参数替代为以参数值为内容的字符窜常量
    ## 将两个相邻的标记(token)连接为一个单独的标记
    #pragma 说明编译器信息
    #warning 显示编译警告信息
    #error 显示编译错误信息
    
    定义常量(用的最多)
    #define M_PI        3.14159265358979323846264338327950288
    
    函数宏(用的较多)
    #define PLUS(x,y) (x + y)//一般建议加括号,因为宏定义只是等名替换
    //下列情况会产生不一样的结果:(两个结果不一样)
    2*PLUS(2,3)=7//如果PLUS(x,y) => x+y
    2*PLUS(2,3)=10//如果PLUS(x,y) =>(x+y)
    

    #######的使用(用的不多)
    #:字符串化操作符。其作用是:将宏定义中的传入参数名转换成用一对双引号括起来参数名字符串(string而不是NSString,所以不能是%@。而是%s)。其只能用于有传入参数的宏定义中,且必须置于宏定义体中的参数名前。
    #define example(instr) printf("the input string is:\t%s\n",#instr)//相当于#define example(instr) printf("the input string is:\t%s\n",“insert”)
    #define example1(instr) #instr //相当于#define example1(instr) “instr”
    编译时则会转换为:

    printf("the input string is:\t%s\n","abc");
    string str="abc";
    

    注意:
    对空格的处理
    a。忽略传入参数名前面和后面的空格。

    如:str=example1( abc ); 将会被扩展成str="abc";

    b.当传入参数名间存在空格时,编译器将会自动连接各个子字符串,用每个子字符串中只以一个空格连接,忽略其中多余一个的空格。

    如:str=exapme( abc ... def); 将会被扩展成 str="abc def";(假设是哪个空格)

    ##用法(用的不多)

    ##的作用则是将宏定义的多个形参成一个实际参数名。
    #define say(n) num##n
    运行:

    int num=say(3);
    int  num3=3;(可能你现在还不知道为什么必须有这个东西,请看注意第二条)
    

    输出:(可见:say(3) ==>num3 )

     num= num3;
    

    注意:
    1.当用##连接形参时,##前后的空格可有可无。
    如:#define exampleNum(n) num ## n
    相当于#define exampleNum(n) num##n

    2.连接后的实际参数名,必须为实际存在的参数名或是编译器已知的宏定义

      // preprocessor_token_pasting.cpp
      #include #define paster( n ) printf_s( "token" #n " = %d", token##n )int token9 = 9;int main(){ paster(9);}
    

    运行结果:
    token9 = 9

    \的用法(用的较多)

    \作用:起到了一个续行的作用,编译器在编译时会忽略行尾的换行符,而把下一行的内容也算作是本行的内容。(常用来定义一个代码块)
    #define GETADDR_SD_MACHSPS(sdAddress)
    AaSysComSicadGet(TASK_DSP_MACHS_PS,
    AaSysComSicadGetNid(sdAddress))

    ...的用法(用的一般)

    printf()和fprintf()这些输出函数的参数是可变的,在调试程序时,你可能希望定义自己的参数可变的输出函数,
    那么可变参数宏会是一个选择。
    C99中规定宏可以像函数一样带有可变参数,比如

    #define LOG(format, ...) fprintf(stdout, format, __VA_ARGS__)
    

    其中,...表示参数可变,VA_ARGS在预处理中为实际的参数集所替换
    GCC中同时支持如下的形式

    #define LOG(format, args...) fprintf(stdout, format, args)
    

    其用法和上面的基本一致,只是参数符号有变化。

    最后一点:Debug通过宏定义过滤掉NSLog(今天实在太晚了,不想写了)

    在项目的过程中,经常遇到要在调试的时候打印log,但是上线或是release 的时候不需要去显示log 的情况,此时你辛辛苦苦写了那么多的log,你要么就手动注释掉,要么就是设一个开关变量,企图用这个总开关开启。其实可以充分利用宏定义进行设置。
    步骤比较简单,只需要 在ProjectName_Prefix.pch 中追加你对应的宏定义,不用import 就可以直接使用了。
    代码如下:
    #ifdef DEBUG
    #define LOG(...) NSLog(VA_ARGS);
    #define LOG_METHOD NSLog(@"%s", func);
    #else
    #define LOG(...); #define LOG_METHOD;
    #endif
    这样设置后,只需要在Product ->Scheme->Edit Scheme ->info选择,是release ,还是debug版本即可。如果debug 版本,则打印log,若release版本则不打印。

    我们发布到appstore 上的版本均是release版本,这里简单说下这两个版本的差异。
    release 是发行版本,比debug版本要小一些,他们调用两个不同底层库,debug 包含的信息多,可以断点调试,单步执行,使用使用TRACE/ASSERT等调试输出语句,
    但是release 版本不包含调试信息,运行速度比较快。
    另外在此处设置的DEBUG 参数可以在下面的路径进行设置:工程->Target->Build Setting ->Preprocessor Macros。默认系统已经给出了DEBUG的参数。如果要增加新的参数,则在哪里进行增加。

    相关文章

      网友评论

        本文标题:宏定义用法的全面探究

        本文链接:https://www.haomeiwen.com/subject/uazklttx.html