iOS 开发中的宏定义

简介

偶然看到群里说iOS适配的时候,给屏幕设定个比例,把这个比例作为宏,我就网上搜了下常用的宏,有些看不懂,仔细研究了下,这篇算是一篇小白文章,我尽量把每一句宏都解释清楚。
一般做iOS开发的时候是建一个pch文件,把常用的宏写在里面,通常网上搜到的都有个#ifndef MacroDefinition_h#define MacroDefinition_h,首先,这个里面是有个#ifndef#define,这个意思我的理解就是没有引用第一个,就引用第二个的意思,往往我们看到的前两行是:

#ifndef MacroDefinition_h
#define MacroDefinition_h

这个意思就是防止MacroDefinition.h被重复引用。
进入正题:

//-------------------获取设备大小-------------------------
//NavBar高度
#define NavigationBar_HEIGHT 44
//获取屏幕 宽度、高度
#define SCREEN_WIDTH ([UIScreen mainScreen].bounds.size.width)
#define SCREEN_HEIGHT ([UIScreen mainScreen].bounds.size.height)

#define是定义的意思,默认就这么写,SCREEN_WIDTH是这个宏的名字,默认全部用大写字母写,后面是代码实现,比如这个就是获取屏幕的宽度。
还有一些复杂的宏定义,先来解释下关键字,原文在此

//关键字
...:可变参数
__VA_ARGS__ :宏定义中的...中的所有剩余参数
##:连接符号
#:原样输出
/:换行符

//系统工具方法
__COUNTER__ 无重复的计数器,从程序启动开始每次调用都会++,常用语宏中定   义无重复的参数名称
__FILE__:当前文件的绝对路径,常见于log中
__LINE__:展开该宏时在文件中的行数,常见于log中
__func__:所在scope的函数名称,常见于log中

举个例子:

/*
获取方法参数的宏bb_argcount()
它的好处不仅将计算在预处理时搞定,不拖延到运行时的cpu;更重要的是编译检查。
如某些可变参数的要求2个或3个参数,其他的都不行。只有这样的宏才能在编译  前就确定参数是否满足要求
参数分类很多步骤,通过不懂的替换宏名称和参数,从而达到计算出方法参数个数的功能。
*/
#define bb_argcount(...) bb_at(10, __VA_ARGS__,10, 9, 8, 7, 6, 5, 4, 3, 2, 1)
#define bb_at(N,...) bb_concat_at##N (__VA_ARGS__)
#define bb_concat_at10(_0,_1,_2,_3,_4,_5,_6,_7,_8,_9,...) bb_head(__VA_ARGS__)
#define bb_head(...) bb_head_first(__VA_ARGS__,0)
#define bb_head_first(first,...) first


//调用示例:
int count = bb_argcount(a,b,c,d,e);
NSLog(@" count is :%d",count);
//输出: count is :5

分析如下:

 bb_argcount(a,b,c,d,e);//假设我们传入5个参数

步骤1:带入bb_argcount
//#define bb_argcount(...) bb_at(10, __VA_ARGS__,10, 9, 8, 7, 6, 5, 4, 3, 2, 1)
//请注意...和__VA_ARGS__的使用,...是可变参数,传入的是a,b,c,d,e会替换__VA_ARGS__的部分,得到如下部分
int count = bb_at(10, a, b, c, d, e, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1)

步骤2:接下来带入bb_at
//#define bb_at(N,...) bb_concat_at##N (__VA_ARGS__)
//第一个参数为N,之后都是可变参数,所以N为10,__VA_ARGS__ 为 a, b, c, d, e, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1
//##是把字符连接起来,通过##我们重新使用了新的宏,这里就是bb_concat_at10
//这一步即修改了参数,又修改了方法名
int count = bb_concat_at10(a, b, c, d, e, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1)

步骤3:带入bbconcatat_10
//bb_concat_at10(_0,_1,_2,_3,_4,_5,_6,_7,_8,_9,...) bb_head(__VA_ARGS__)
//把前面10个参数都换了,第10位以后的参数设为可变参数
int count = bb_head(5,4,3, 2, 1)

步骤4:带入bb_head
//bb_head(first,...) first
//直接获取第一个数,其他的省略
int count = 5;

还有一篇入门的文章,写的很好,可以看看。
接下来就是一些常用的宏了:

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

#ifdef DEBUG

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

#else

#define DLog(...)

#endif

//重写NSLog,Debug模式下打印日志和当前行数

//#if DEBUG

//#define NSLog(FORMAT, ...) fprintf(stderr,"\nfunction:%s line:%d content:%s\n", __FUNCTION__, __LINE__, [[NSString stringWithFormat:FORMAT, ##__VA_ARGS__] UTF8String]);

//#else

//#define NSLog(FORMAT, ...) nil

//#endif

//DEBUG模式下打印日志,当前行并弹出一个警告

#ifdef DEBUG

#define ULog(fmt, ...){ UIAlertView *alert = [[UIAlertView alloc]\

initWithTitle:[NSString stringWithFormat:@"%s\n [Line %d] ", __PRETTY_FUNCTION__, __LINE__] essage:[NSString stringWithFormat:fmt, ##__VA_ARGS__]delegate:nil cancelButtonTitle:@"Ok"otherButtonTitles:nil]; [alert show]; }

#else

#define ULog(...)

#endif
#define ITTDEBUG

#define ITTLOGLEVEL_INFO10

#define ITTLOGLEVEL_WARNING3

#define ITTLOGLEVEL_ERROR1

#ifndef ITTMAXLOGLEVEL

#ifdef DEBUG

#define ITTMAXLOGLEVEL ITTLOGLEVEL_INFO

#else

#define ITTMAXLOGLEVEL ITTLOGLEVEL_ERROR

#endif

#endif

// The general purpose logger. This ignores logging levels.

#ifdef ITTDEBUG

#define ITTDPRINT(xx, ...)NSLog(@"%s(%d): "xx, __PRETTY_FUNCTION__, __LINE__, ##__VA_ARGS__)

#else

#define ITTDPRINT(xx, ...)((void)0)

#endif

// Prints the current method's name.

#define ITTDPRINTMETHODNAME() ITTDPRINT(@"%s", __PRETTY_FUNCTION__)

// Log-level based logging macros.

#if ITTLOGLEVEL_ERROR <= ITTMAXLOGLEVEL

#define ITTDERROR(xx, ...)ITTDPRINT(xx, ##__VA_ARGS__)

#else

#define ITTDERROR(xx, ...)((void)0)

#endif

#if ITTLOGLEVEL_WARNING <= ITTMAXLOGLEVEL

#define ITTDWARNING(xx, ...)ITTDPRINT(xx, ##__VA_ARGS__)

#else

#define ITTDWARNING(xx, ...)((void)0)

#endif

#if ITTLOGLEVEL_INFO <= ITTMAXLOGLEVEL

#define ITTDINFO(xx, ...)ITTDPRINT(xx, ##__VA_ARGS__)

#else

#define ITTDINFO(xx, ...)((void)0)

#endif

#ifdef ITTDEBUG

#define ITTDCONDITIONLOG(condition, xx, ...) { if ((condition)) { \

ITTDPRINT(xx, ##__VA_ARGS__); \

} \

} ((void)0)

#else

#define ITTDCONDITIONLOG(condition, xx, ...) ((void)0)

#endif

#define ITTAssert(condition, ...)\

do {\

if (!(condition)) {\

[[NSAssertionHandler currentHandler]\

handleFailureInFunction:[NSString    stringWithUTF8String:__PRETTY_FUNCTION__] \

file:[NSString stringWithUTF8String:__FILE__]\

lineNumber:__LINE__\

description:__VA_ARGS__];\

}\

} while(0)

//----------------------系统----------------------------

//获取系统版本

#define IOS_VERSION [[[UIDevice currentDevice] systemVersion] floatValue]
#define CurrentSystemVersion [[UIDevice currentDevice] systemVersion]

//获取当前语言

#define CurrentLanguage ([[NSLocale preferredLanguages] objectAtIndex:0])

//判断是否Retina屏、设备是否%fhone 5、是否是iPad

#define isRetina ([UIScreen instancesRespondToSelector:@selector(currentMode)] ?\

CGSizeEqualToSize(CGSizeMake(640,960), [[UIScreen mainScreen] currentMode].size) : NO)

#define iPhone5 ([UIScreen instancesRespondToSelector:@selector(currentMode)] ? \

CGSizeEqualToSize(CGSizeMake(640,1136), [[UIScreen mainScreen] currentMode].size) : NO)

#define isPad (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad)

//判断是真机还是模拟器

#if TARGET_OS_IPHONE

//iPhone Device

#endif
#if TARGET_IPHONE_SIMULATOR

//iPhone Simulator

#endif

//检查系统版本

#define SYSTEM_VERSION_EQUAL_TO(v)([[[UIDevice currentDevice] systemVersion]\

compare:v options:NSNumericSearch] == NSOrderedSame)

#define SYSTEM_VERSION_GREATER_THAN(v)([[[UIDevice currentDevice] systemVersion] \

compare:v options:NSNumericSearch] == NSOrderedDescending)

#define SYSTEM_VERSION_GREATER_THAN_OR_EQUAL_TO(v)([[[UIDevice currentDevice] systemVersion]\

compare:v options:NSNumericSearch] != NSOrderedAscending)

#define SYSTEM_VERSION_LESS_THAN(v)([[[UIDevice currentDevice] systemVersion]\

compare:v options:NSNumericSearch] == NSOrderedAscending)

#define SYSTEM_VERSION_LESS_THAN_OR_EQUAL_TO(v)([[[UIDevice currentDevice] systemVersion]\

compare:v options:NSNumericSearch] != NSOrderedDescending)

//----------------------系统----------------------------

//----------------------内存----------------------------

//使用ARC和不使用ARC

#if __has_feature(objc_arc)

//compiling with ARC

#else

// compiling without ARC

#endif
#pragma mark - common functions
#define RELEASE_SAFELY(__POINTER) { [__POINTER release]; __POINTER = nil; }

//释放一个对象

#define SAFE_DELETE(P) if(P) { [P release], P = nil; }
#define SAFE_RELEASE(x) [x release];x=nil

//----------------------内存----------------------------

//----------------------图片----------------------------

//读取本地图片

#define LOADIMAGE(file,ext) [UIImage imageWithContentsOfFile:[[NSBundle mainBundle]pathForResource:file ofType:ext]]

//定义UIImage对象

#define IMAGE(A) [UIImage imageWithContentsOfFile:[[NSBundle mainBundle] pathForResource:A ofType:nil]]

//定义UIImage对象

#define ImageNamed(_pointer) [UIImage imageNamed:[UIUtil imageName:_pointer]]

//建议使用前两种宏定义,性能高于后者

//----------------------图片----------------------------

//----------------------颜色类---------------------------

// rgb颜色转换(16进制->10进制)

#define UIColorFromRGB(rgbValue) \

[UIColor colorWithRed:((float)((rgbValue &0xFF0000) >>16))/255.0\

green:((float)((rgbValue &0xFF00) >>8))/255.0\

blue:((float)(rgbValue &0xFF))/255.0alpha:1.0]

//带有RGBA的颜色设置

#define COLOR(R, G, B, A) [UIColor colorWithRed:R/255.0green:G/255.0blue:B/255.0alpha:A]

//获取RGB颜色

#define RGBA(r,g,b,a) [UIColor colorWithRed:r/255.0fgreen:g/255.0fblue:b/255.0falpha:a]
#define RGB(r,g,b) RGBA(r,g,b,1.0f)

//背景色

#define BACKGROUND_COLOR [UIColor colorWithRed:242.0/255.0green:236.0/255.0blue:231.0/255.0alpha:1.0]

//清除背景色

#define CLEARCOLOR [UIColor clearColor]
#pragma mark - color functions
#define RGBCOLOR(r,g,b) [UIColor colorWithRed:(r)/255.0fgreen:(g)/255.0fblue:(b)/255.0falpha:1]
#define RGBACOLOR(r,g,b,a) [UIColor colorWithRed:(r)/255.0fgreen:(g)/255.0fblue:(b)/255.0falpha:(a)]

//----------------------颜色类--------------------------

//----------------------其他----------------------------

//方正黑体简体字体定义

#define FONT(F) [UIFont fontWithName:@"FZHTJW--GB1-0"size:F]

//定义一个API

#define APIURL@"http://xxxxx/"

//登陆API

#define APILogin[APIURL stringByAppendingString:@"Login"]

//设置View的tag属性

#define VIEWWITHTAG(_OBJECT, _TAG)[_OBJECT viewWithTag : _TAG]

//程序的本地化,引用国际化的文件

#define MyLocal(x, ...) NSLocalizedString(x, nil)

//G-C-D

#define BACK(block) dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT,0), block)
#define MAIN(block) dispatch_async(dispatch_get_main_queue(),block)

//NSUserDefaults实例化

#define USER_DEFAULT [NSUserDefaults standardUserDefaults]

//由角度获取弧度有弧度获取角度

#define degreesToRadian(x) (M_PI * (x) /180.0)
#define radianToDegrees(radian) (radian*180.0)/(M_PI)
#define LITE_RESOURCE_BUNDLE[[ NSBundle mainBundle] pathForResource:\

@"LiteResource"\

ofType:@"bundle"];

//加锁

#define onExit \

__strong void (^block)(void) attribute((cleanup(blockCleanUp),unused)) = ^

/*

NSRecursiveLock *aLock = [[NSRecursivewLock alloc] init];

[aLock lock];

onExit {

[aLock unlock];//妈妈再也不用担心我忘写后半段了

}

*/

//TODO宏定义

#define STRINGIFY(S) #S
#define DEFER_STRINGIFY(S) STRINGIFY(S)
#define PRAGMA_MESSAGE(MSG) _Pragma(STRINGIFY(message(MSG)))
#define FORMATTED_MESSAGE(MSG)"[TODO-"DEFER_STRINGIFY(COUNTER)"] "MSG" \n"\

DEFER_STRINGIFY(FILE)" line "DEFER_STRINGIFY(LINE)

#define KEYWORDIFY try {} @catch (...) {}

//最终使用下面的宏

#define TODO(MSG) KEYWORDIFY PRAGMA_MESSAGE(FORMATTED_MESSAGE(MSG))
#endif

GCD 与 #define

#define DISPATCH_ONCE_BLOCK(onceBlock) static dispatch_once_t onceToken; dispatch_once(&onceToken, onceBlock);
  • (instancetype)sharedService {staticXXX *sharedService = nil;//直接一行搞定DISPATCH_ONCE_BLOCK(^{ sharedService = [XXXnew]; })returnsharedService;}

//在Main线程上运行#defineDISPATCH_ON_MAIN_THREAD(mainQueueBlock)dispatch_async(dispatch_get_main_queue(),mainQueueBlock);//在Global Queue上运行

#defineDISPATCH_ON_GLOBAL_QUEUE_HIGH(globalQueueBlocl)dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH,0),globalQueueBlocl);
#defineDISPATCH_ON_GLOBAL_QUEUE_DEFAULT(globalQueueBlocl)dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT,0),globalQueueBlocl);
#defineDISPATCH_ON_GLOBAL_QUEUE_LOW(globalQueueBlocl)dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_LOW,0),globalQueueBlocl);
#defineDISPATCH_ON_GLOBAL_QUEUE_BACKGROUND(globalQueueBlocl)dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND,0),globalQueueBlocl);

//主线程上DISPATCH_ON_MAIN_THREAD(^{

//更新UI

})

//Global QueueDISPATCH_ON_GLOBAL_QUEUE_DEFAULT(^{

//异步耗时任务

})

//获取View的属性

#defineGetViewWidth(view)view.frame.size.width
#defineGetViewHeight(view)view.frame.size.height
#defineGetViewX(view)view.frame.origin.x
#defineGetViewY(view)view.frame.origin.y

//屏幕常量

#define GetScreenWidth [[UIScreen mainScreen] bounds].size.width
#define GetScreenHeight [[UIScreen mainScreen] bounds].size.height

#defineSuppressPerformSelectorLeakWarning(Stuff)do{_Pragma("clang diagnostic push")_Pragma("clang diagnostic ignored"-Warc-performSelector-leaks"")Stuff;_Pragma("clang diagnostic pop")}while(0)

_Pargma的处理方式是把字符串常量的内容(在删除两边的双引号,并把字符串常量内部的\“替换为”,把\替换为\之后)看成是#pragma指令中出现的预处理器标记。 这段代码的基本流程:

  1. push 当前警告入栈

  2. 忽略我们要消除的警告

  3. 执行会产生警告的代码

  4. pop 警告出栈——恢复之前的状态
    参考文章:
    [http://www.2cto.com/kf/201512/454887.html]
    [http://www.jianshu.com/p/dde321552fdc]
    [http://blog.csdn.net/duxinfeng2010/article/details/9067947]
    [http://www.cocoachina.com/ios/20160713/17026.html]
    [http://blog.csdn.net/li_yangyang_li/article/details/50611971]
    [http://blog.csdn.net/hopedark/article/details/20699723]

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,847评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,208评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,587评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,942评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,332评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,587评论 1 218
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,853评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,568评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,273评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,542评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,033评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,373评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,031评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,073评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,830评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,628评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,537评论 2 269

推荐阅读更多精彩内容