宏定义用法的全面探究

废话

由于昨天一直在学习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的参数。如果要增加新的参数,则在哪里进行增加。

欢迎关注我的个人微信公众号,免费送计算机各种最新视频资源!你想象不到的精彩!


宏定义用法的全面探究_第1张图片
0.jpg

你可能感兴趣的:(宏定义用法的全面探究)