编码篇-学会小用宏和条件编译

前言

宏定义在C系开发中可以说占有举足轻重的作用。底层框架自不必说,为了编译优化和方便,以及跨平台能力,宏被大量使用,可以说底层开发离开define将寸步难行。而在更高层级进行开发时,我们会将更多的重心放在业务逻辑上,似乎对宏的使用和依赖并不多。 编译时编译器会在语义分析认定是宏后,将形参替换为实参,这个过程称为宏的展开。

使用宏的好处:

  • 在节省工作量的同时,代码可读性大大增加。如打印语句可以使打印出来的内容更美观。
  • 写出漂亮优雅的代码(虽然宏本身可能并不漂亮优雅)。
  • 我们使用宏一是为了多处使用方便,而是方便修改:一处修改全局适用。

宏的分类使用

一般宏分为两类 对象宏(object-like macro)和函数宏(function-like macro)。

  • 对象宏:对于对象宏来说确实相对简单,一般用来定义一些常数。
  • 函数宏:函数宏顾名思义,就是行为类似函数,可以接受参数的宏。 在书写函数宏的时候注意多加(),这样可以避免优先级所造成的问题

一般开发中使用的宏大体为:

  • 颜色
  • 服务器地址
  • 储存取值
  • 屏幕宽高
  • 导航高度
  • 缩放比例
  • 其他常用的代码段

常用的宏定义(不全,但是以此类推)

#define ScreenWidth  [UIScreen mainScreen].bounds.size.width
#define ScreenHeight [UIScreen mainScreen].bounds.size.height

#define NavBarHeight 64.0f
#define TabBarHeight 49.0f

#define selfWidth     self.frame.size.width
#define selfHeight    self.frame.size.height

#define ViewBorderRadius(View,Radius,Width,Color)    [View.layer setCornerRadius:(Radius)];\
                                                       [View.layer setMasksToBounds:YES];\
                                                       [View.layer setBorderWidth:(Width)];\
                                                       [View.layer setBorderColor:[Color CGColor]];//设置圆角

weak弱引用self

  #define WeakSelf __weak typeof(self) weakSelf = self;

weak弱引用对象

  #define WeakObj(obj) __weak typeof(obj) weakObj = obj;

strong强引用self

  #define StrongSelf(weakSelf) __strong typeof(weakSelf) strongSelf = weakSelf;

strong强引用对象

  #define StrongObj(obj) @autoreleasepool{} __strong typeof(obj) obj = strongObj;


  #define IS_IOS8       ([[[UIDevice currentDevice] systemVersion] floatValue] >= 8)

  //判断是不是iPad
  #define IS_iPad        UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad
  //判断是不是iphone
  #define IS_IPHONE      [[UIDevice currentDevice] userInterfaceIdiom] == UIUserInterfaceIdiomPhone


  #define GetImage(imageName)     [UIImage imageNamed:[NSString stringWithFormat:@"%@",imageName]]

  #define NEWSTITLECOLOR ([UIColor colorWithRed:64/255.0 green:56/255.0 blue:53/255.0 alpha:1.0f])

  #define showMessage(a)   [SVProgressHUD showInfoWithStatus:a maskType:SVProgressHUDMaskTypeGradient];
  #define showSuccessMessage(a)  [SVProgressHUD showSuccessWithStatus:a maskType:SVProgressHUDMaskTypeGradient];

  #define SandboxPathStr [NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) firstObject]
  #define cafPathStr     [SandboxPathStr stringByAppendingPathComponent:@"myRecord.caf"]
  #define mp3PathStr     [SandboxPathStr stringByAppendingPathComponent:@"myRecord.mp3"]

字符串是否为空

  #define LMStringIsEmpty(str)   ([str isKindOfClass:[NSNull class]] || str == nil || [str length] < 1 ? YES : NO )

数组是否为空

  #define LMArrayIsEmpty(array)  (array == nil || [array isKindOfClass:[NSNull class]] || array.count == 0)

字典是否为空

  #define LMDictIsEmpty(dic) (dic == nil || [dic isKindOfClass:[NSNull class]] || dic.allKeys == 0)

  #define LMApplication        [UIApplication sharedApplication]
  #define LMKeyWindow          [UIApplication sharedApplication].keyWindow

  #define LMAppVersion         [[[NSBundle mainBundle] infoDictionary] objectForKey:@"CFBundleShortVersionString"]

property属性快速声明

#define PropertyString(s)                @property(nonatomic,copy)NSString    * s   
#define PropertyNSInteger(s)             @property(nonatomic,assign)NSInteger  s
#define PropertyFloat(s)                 @property(nonatomic,assign)float  s
#define   propertyStrong(a,b)            @property (strong, nonatomic) a *b;

DEBUG模式下打印日志,当前行

#ifdef DEBUG
    #define DLog(fmt,...)NSLog((@"%s[Line %d]" fmt),__PRETTY_FUNCTION__,__LINE__,##__VA_ARGS__);
#else
    #define DLog(...)
#endif

传参view对象,获取view的frame、bounds相关属性值

#define VIEW_BOUNDS(aView)       ((aView).bounds)

#define VIEW_FRAME(aView)        ((aView).frame)

#define VIEW_ORIGIN(aView)       ((aView).frame.origin)
#define VIEW_X(aView)            ((aView).frame.origin.x) 
#define VIEW_Y(aView)            ((aView).frame.origin.y)

#define VIEW_SIZE(aView)         ((aView).frame.size)
#define VIEW_HEIGHT(aView)       ((aView).frame.size.height)  // 视图高度
#define VIEW_WIDTH(aView)        ((aView).frame.size.width)   // 视图宽度

获取iPhone屏幕尺寸

// x
#define IS_iPhoneX        ([UIScreen instancesRespondToSelector:@selector(currentMode)] ? CGSizeEqualToSize(CGSizeMake(1125, 2436), [[UIScreen mainScreen] currentMode].size) : NO)
// 6P、6sP、7P、8P
#define IS_iPhone678_Plus ([UIScreen instancesRespondToSelector:@selector(currentMode)] ? CGSizeEqualToSize(CGSizeMake(1242, 2208), [[UIScreen mainScreen] currentMode].size) : NO)
// 6、6s、7、8
#define IS_iPhone678      ([UIScreen instancesRespondToSelector:@selector(currentMode)] ? CGSizeEqualToSize(CGSizeMake(750, 1334), [[UIScreen mainScreen] currentMode].size) : NO)
// 5、5s
#define IS_iPhone5        ([UIScreen instancesRespondToSelector:@selector(currentMode)] ? CGSizeEqualToSize(CGSizeMake(640, 1136), [[UIScreen mainScreen] currentMode].size) : NO)
// 4、4s
#define IS_iPhone4       ([UIScreen instancesRespondToSelector:@selector(currentMode)] ? CGSizeEqualToSize(CGSizeMake(640, 960), [[UIScreen mainScreen] currentMode].size) : NO)

大小屏字体自动切换

有的应用希望有一个好的用户体验会在不同的屏幕上适配不同大小字体,这时就可以使用以下的宏定义来实现。但是如果应用中字体大小不能做到全局统一,就不要使用以下的宏定义来实现字体大小适配。例如:

#define IS_SmallScreen (IS_iPhone5 || IS_iPhone4)

#define MaxFontSize    (IS_SmallScreen ? 21.f : 25.f )
#define LagerFontSize  (IS_SmallScreen ? 17.f : 19.f )
#define BigFontSize    (IS_SmallScreen ? 15.f : 17.f )
#define NormalFontSize (IS_SmallScreen ? 13.f : 15.f )
#define SmallFontSize  (IS_SmallScreen ? 11.f : 13.f )
#define MinFontSize    (IS_SmallScreen ? 9.f  : 11.f )

校验相关

#define IsCanUseString(str)     ((str != nil) && ![str isKindOfClass:[NSNull class]] && [str isKindOfClass:[NSString class]] && [str length] > 0 )
#define IsCanUseArray(arr)      ( arr && (arr != nil) && ![arr isKindOfClass:[NSNull class]] )
#define IsCanUseDic(dic)        ( dic && (dic != nil) && ![dic isKindOfClass:[NSNull class]] )
#define IsCanUseObj(obj)        ( obj && (obj != nil) && ![obj isKindOfClass:[NSNull class]] )
#define IsNullClass(class)      [class isKindOfClass:[NSNull class]]

打印相关

这样的打印语句,省事而且美观易读
// mark(NSString类型参数)为打印内容标题

#define NSLOG_Str(mark,str)       NSLog(@"##%@##--str:%@--",(mark),(str))
#define NSLOG_Int(mark,int)       NSLog(@"##%@##--int:%ld--",(mark),(int))
#define NSLOG_Float(mark,float)   NSLog(@"##%@##--float:%f--",(mark),(float))
#define NSLOG_Bool(mark,bool)     NSLog(@"##%@##--bool:%@--",(mark),(bool) ? @"YES" : @"NO")
#define NSLOG_Point(mark,point)   NSLog(@"##%@##--x:%f--y:%f--",(mark),(point).x,(point).y)
#define NSLOG_Size(mark,size)     NSLog(@"##%@##--width:%f--height:%f--",(mark),(size).width,(size).height)
#define NSLOG_Frame(mark,frame)   NSLog(@"##%@##--x:%f--y:%f--width:%f--height:%f--",(mark),(frame).origin.x,(frame).origin.y,(frame).size.width,(frame).size.height)

条件编译

一般情况下,源程序中所有的行都参加编译。但是有时希望对其中一部分内容只在满足一定条件才进行编译,也就是对一部分内容指定编译的条件,这就是条件编译(不被编译的代码不会被运行)

条件编译语法格式

1、#if 编译预处理中的条件命令, 相当于C语法中的if语句 2、#ifdef 判断某个宏是否被定义, 若已定义, 执行随后的语句 3、#ifndef 与#ifdef相反, 判断某个宏是否未被定义 4、#elif 若#if, #ifdef, #ifndef或前面的#elif条件不满足, 则执行#elif之后的语句, 相当于C语法中的else-if 5、#else 与#if, #ifdef, #ifndef对应, 若这些条件不满足, 则执行#else之后的语句, 相当于C语法中的else 6、#endif #if, #ifdef, #ifndef这些条件命令的结束标志. 7、#if 与 #ifdef 的区别:#if是判断后面的条件语句是否成立,#ifdef是判断某个宏是否被定义过。要区分开!

 #if 条件1
 ...code1...
 #elif 条件2
  ...code2...
 #else
 ...code3...
 #endif
 *******************
  #ifdef 标识符    // 如果定义了 标识符
    程序段1
 #else
     程序段2
 #endif
 它的作用是:当标识符已经被定义过(一般是用#define命令定义),则对程序段1进行编译,否则编译程序段2。其中#else部分也可以没有,即:
  #ifdef
      程序段1
  #denif
*******************
  #ifndef MACRO_Define // 如果未定义MACRO_Define这个宏
      代码块1
  #else
      代码块2
  #endif

说明:预处理指令是编译之前的,不是运行时的,所以条件编译时要注意if的条件,不要还没运行,就先用源程序里面的变量作为条件进行判断,变量是运行时才产生的,而条件编译呢是在运行之前编译的。所以条件编译的条件一般是利用宏定义,因为宏定义和条件编译都是编译之前进行的。

如下面的一个错误例子:

  #include<stdio.h>
   void main()
  {
      int a =8;
      #if a>7
          printf("a>7");
      #elif a<7
          printf("a<7");
    #else
        printf("a!=7");
    #endif
}
运行结果为a<7,与我们期望的结果不一样,正常应该是输出a>7才对。

正确的写法如下:

 #include<stdio.h>
 #define a 8
  void main()
  {
    #if a>7
        printf("a>7");
    #elif a<7
      printf("a<7,%d",a);
    #else
        printf("a!=7");
    #endif
  }
  输出结果为a>7

条件编译的使用

测试服务器、正式服务器的自动切换。

#ifdef DEBUG
    #define request11   @"123"
#else
    #define request11   @"4565"
#endif

如何设置环境变量配置进行条件编译

  • 通过 Configurations 添加多个环境
  • 不同环境下设置不同的宏定义(在某个环境下设置的宏只能在哪个环境下的 Targets 中被识别,否则会报错,所以建议不同环境下定义同一个全局变量为不同的值。)
  • 区分不同的环境 if (DEBUG==1) { NSLog(@"测试环境"); }else if (DEBUG==2){ NSLog(@"运营环境"); }else{ NSLog(@"生产环境"); }

我们可以通过设置多个环境,每一种环境下的值不同; 生成多个Scheme,每一个Scheme对应一种环境和配置; 这样切换Scheme 即可切换到不同的环境下。

感兴趣的可以查看我的另一篇文章:基础篇-工程管理之多Targets

小结

宏的使用,重点是函数宏的使用,后续有新的使用心得会持续更新本文。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Java学习网

常见的 Java 错误及避免方法之第五集(每集10个错误后续持续发布)

当输入期间意外终止文件或流时,将抛出“EOFException”。 以下是抛出EOFException异常的一个示例,来自JavaBeat应用程序:

1343
来自专栏跟着阿笨一起玩NET

以读取博客园随笔备份为例 将xml 序列化成json,再序列化成对象

资源下载:http://files.cnblogs.com/codealone/ConsoleApplication2.zip

761
来自专栏Java架构师历程

【datatable】Cannot read property ‘style’ of undefined问题解决

遇到这个问题的时候一开始我以为是引入的js有问题,后来研究了源码之后原来是datatable的列的数量和表头列的数量没有对齐,表头我定义的列是这样的

1851
来自专栏刘望舒

LeakCanary看这一篇文章就够了

LeakCanary是Square公司基于MAT开源的一个内存泄漏检测工具,在发生内存泄漏的时候LeakCanary会自动显示泄漏信息。

1K5
来自专栏Android 开发学习

JsBridge 源码分析

1573
来自专栏大内老A

Enterprise Library深入解析与灵活应用(6):自己动手创建迷你版AOP框架

基于Enterprise Library PIAB的AOP框架已经在公司项目开发中得到广泛的使用,但是最近同事维护一个老的项目,使用到了Enterprise L...

2068
来自专栏青蛙要fly的专栏

项目需求讨论-Vlayout来快速构建及扩展复杂界面

大家好,今天又带来了项目中具体遇到的需求。做一个首界面,该首界面有很多功能块,同时这些功能块是动态的,因为登录的人的权限的不同,会显示不同的功能块,因为功能模块...

1812
来自专栏大内老A

ASP.NET MVC的Razor引擎:RazorView

Razor引擎具有两个核心的类型,一个是表示View本身的类型RazorView,另一个则是获取和创建它的RazorViewEngine,我们将用两篇文章对它们...

2367
来自专栏逸鹏说道

我这么玩Web Api(二)

数据验证,全局数据验证与单元测试 目录 一、模型状态 - ModelState 二、数据注解 - Data Annotations 三、自定义数据注解 四、全局...

4966
来自专栏Java架构沉思录

深入理解-Spring-之源码剖析IOC(一)

作为Java程序员,Spirng我们再熟悉不过,可以说比自己的女朋友还要亲密,每天都会和他在一起,然而我们真的了解spring吗?

973

扫码关注云+社区