华丽的#define, 预编译介绍~

一般情况下,我们使用#define来定义一个常量,#define的本质是文本替换,例如#define INT_PTR int*,这时候我们使用INT_PTR,INT_PTR a,b;这条语句等价于int * a,b;也就是定义了一个指针变量a和整型变量b,这是#define常用的场景和需要注意的细节地方。下面我收集并整理了常用的#define,以后也会不断地更新。

1、定义常量

定义常量的时候最好以小写字母k开头,让人见名知意,

(1)导航栏高度:我们都知道iPhone竖屏时候导航栏的高度为44,这时候可以定义一个常量来表示该高度,

#define kNaivgationBarHeight 44

(2)屏幕的宽高:屏幕的宽高就是iOS设备硬件的屏幕尺寸,跟ViewController的view不完全相同,

#define kSCREEN_WIDTH [UIScreen mainScreen].bounds.size.width

#define kSCREEN_HEIGHT [UIScreen mainScreen].bounds.size.height

2、内存管理的安全释放对象

#define SAFE_RELEASE(x) [x release];x=nil

注意结尾没有;冒号,这条语句在dealloc时候使用,例如

- (void)dealloc

{

SAFE_RELEASE(array);

[super dealloc];

}

为什么这句话表示安全释放呢?我们在使用Objective-C对象的时候,最后一定要保证它的引用计数retainCount为0,但是有时候我们也不能完全保证自己做到完美,这时候在dealloc的时候将对象设置为nil,这样就释放了该对象战友的内存区域,防止内存泄露。

3、判断iOS系统的版本

(1)当前系统版本号

#define kCurrentSystemVersion [[[UIDevice currentDevice] systemVersion] floatValue]
(2)判断是否是iOS7或更高的系统版本

#define IOS_VERSION_7_OR_LATER (([[[UIDevice currentDevice] systemVersion] floatValue] >= 7.0)? (YES):(NO))
(3)当前的系统语言

#define kCurrentLanguage [[NSLocale preferredLanguages] objectAtIndex:0]
4、定义常用的颜色

有时候多个控件都需要设置同一个颜色,而UIColor的rgb写法确实浪费时间,是用宏定义常量,可以节省很多的代码,例如下面定义了紫色和暗灰,

#define kPurpleColor [UIColor colorWithRed:137.0/255 green:21.0/255 blue:89.0/255 alpha:1.0]
#define kDarkGrayColor [UIColor colorWithRed:100.0/255 green:100.0/255 blue:100.0/255 alpha:1.0]
这时候给控件定义背景色就方便多了

5、定义比NSLog更高级的DLog

NSLog方便我们暴力调试,就是输出自己观察的变量的值,是用宏定义可以将NSLog封装得更加高级,在项目的pch文件中,是用如下代码,

#define DEBUG_MODE 1
#if DEBUG_MODE
#define DLog( s, ... ) NSLog( @" %@", self, [[NSString stringWithUTF8String:__FILE__] lastPathComponent], __LINE__, [NSString stringWithFormat:(s), ##__VA_ARGS__] )
#else
#define DLog( s, ... )
#endif
例如在ViewController中使用DLog(@"12345");在控制台上面输出的内容如下,

2014-04-18 19:33:30.377 DefineSample[3593:70b] 12345
这段信息包括字符串@"12345"的内存地址,所在的文件ViewController.m的54行,字符串内容为12345。其实看一看这些宏的定义,我们可以了解的更多系统的东西,例如__FILE__表示定位到哪个文件,__LINE__定位到哪一行。

当我们在Debug项目的时候,会产生这些输出;我们发布(release)项目的时候,将#define DEBUG_MODE 1注释掉,这时候就不会产生输出了,毕竟输出也是要耗费CPU资源,降低APP运行效率,虽然影响微乎其微,但是程序员做事就是要精细嘛。这种手动配置项目的方法,熟练是用可以极大地提高开发效率。需要注意的是这个宏只能将NSString作为参数,输出NSString的内容,对于数组、字典、UI控件以及基本类型int、float则不能作为其参数。不过可以自己去定义需要的宏,将上述的类型作为参数,也不是很困难。

6、判断是iPhone真机(Device)还是模拟器(Simulator)

#if TARGET_OS_IPHONE
//针对真机进行编码
NSLog(@"iPhone Device");
#elif TARGET_IPHONE_SIMULATOR
//针对模拟器编码
NSLog(@"iPhone Simulator");
#endif
有的时候模拟器和真机的性能不一样,所以这样可以做一个判断。上面的宏TARGET_OS_IPHONE和TARGET_IPHONE_SIMULATOR是系统定义的,可以直接是用,按住Command点击,可以看见更多的信息。

7、判断是否是ARC

//ARC

#if __has_feature(objc_arc)

//是用arc编码

#else

//是用手动内存管理

#endif

8、定义GCD的后台线程和主线程

//后台运行

#define BACK_GCD(block) dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), block)

//主线程运行
#define MAIN_GCD(block) dispatch_async(dispatch_get_main_queue(),block)

9、单例化一个类

#define SYNTHESIZE_SINGLETON_FOR_CLASS(classname) static classname *shared##classname = nil; + (classname *)shared##classname { @synchronized(self) { if (shared##classname == nil) { shared##classname = [[self alloc] init]; } } return shared##classname; } + (id)allocWithZone:(NSZone *)zone { @synchronized(self) { if (shared##classname == nil) { shared##classname = [super allocWithZone:zone]; return shared##classname; } } return nil; } - (id)copyWithZone:(NSZone *)zone { return self; }
注意这是arc时候单例一个类的宏,另外一个注意的地方就是上面是用了‘\‘反斜杠,这是#define时候换行的时候要在行末加上换行,不然相当于连成一块的字符串。

预处理过程扫描源代码,对其进行初步的转换,产生新的源代码提供给编译器。可见预处理过程先于编译器对源代码进行处理。
预处理指令是以#开头的代码行,#后是指令关键字,在关键字和#号之间允许存在任意个数的空白字符。正行语句构成了一条预处理指令,该指令酱紫啊编译器进行编译之前对源代码做某些转换,下面是一些常用的预处理指令,
# 空指令,没有任何效果
#include 包含一个源代码文件
#define 定义宏
#undef 取消定义宏
#if 如果条件为真,则编译下面的代码
#elif 如果前面的#if不为真,则编译下面的代码
#endif 结束一个#if...#elif条件编译块
#ifdef 如果已经定义了某个宏,则编译下面的代码
#ifndef 如果没有定义某个宏,则编译下面的代码
#error 停止编译并显示错误信息

最近在看公司自己封装的sdk的时候,遇到了很多#ifdef和#ifndef,下面我就主要距离来说一说这个预处理指令的灵活使用。比如你现在通过代码创建了一个Button,
UIButton *btn = [UIButton buttonWithType:UIButtonTypeRoundedRect];
btn.frame = CGRectMake(0, 0, 100, 40);
[btn setTitle:@"BUTTON" forState:UIControlStateNormal];
[btn addTarget:self action:@selector(btnClicked) forControlEvents:UIControlEventTouchUpInside];
[self.view addSubview:btn];
但是根据新的需求,这个Button暂时不需要显示在界面上面,这时候不可以将该Button相关的代码注释掉,这也不失为一个好方法。但是我更推荐使用预处理指令来进行条件判断,下面我使用两种方法来说明,
(1)第一种方法,定义一个具体的宏
#define Debug_ShowButton 0
{
//创建Button对象btn
......//省略若干代码
#if Debut_ShowButton
//如果需要显示按钮,那么将其添加到父视图中
[self.view addSubview:btn];
#else
//不将其添加到父视图中
//[self.view addSubview:btn];
#endif
}
这时候如果想显示该Button对象,那么将#define Debug_ShowButton 1,否则写为#define Debug_ShowButton 0。
(2)第二种方法,定义一个没有数值的宏
{
#ifdef Debug_ShowButton
[self.view addSubview:btn];
#endif
}
这段代码的意思就是如果定义了宏Debug_ShowButton,那么就将btn添加到父视图,如果没有定义的话,那么不添加到父视图。这时候如果我们想要将其添加到父视图,则在文件的顶部写上#define Debug_ShowButton,对,这时候我们可以不给这个宏指定对应的值,这个代码仅仅表示定义一个没有对应值的宏。
总结:
这两种方法,我更加推荐第二种,因为一般情况下,我们定义一个有对应数值的宏,是为了在代码中使用该数值,例如#define kStatusBarHieght 20,这样我们就可以把kStatusBarHeight作为数值在代码中使用,而第一种方法定义了有对应值的宏Debug_ShowButton,我们却没有在代码中使用它,这样是不是有点浪费呢;而使用第二种方法定义一个没有对应值的宏,它只是一个标识符,看起来简单点。我个人更加推荐第二种!

----------------------------------摘自csdn

时间: 2024-08-10 21:20:55

华丽的#define, 预编译介绍~的相关文章

#define(宏替换)以及如何打开预编译之后的“.i”文件看宏的本质

<span style="font-size:24px;">#include<stdio.h> #include<iostream.h> #define PI 3.14 /* <span style="color:#ff0000;">PI是符号常量,不开辟空间,只是个临时符号 宏的本质是:替换</span> */ int main() { double result; int r = 3; result =

C#中的预编译指令介绍

原文:C#中的预编译指令介绍 1.#define和#undef 用法: #define DEBUG #undef DEBUG #define告诉编译器,我定义了一个DEBUG的一个符号,他类似一个变量,但是它没有具体的值,可以将它看为一个符号而已.#undef就是删除这个符号的定义.如果符号DEBUG没定义过,则#undef不起作用,否则#define不起作用.二者都必须放在源代码之前.二者的顺序看代码的顺序: #define DEBUG #undef  DEBUG 这样的话,DEBUG是没有定

预编译那些事#define后面只跟一个“参数”

一般情况下,源程序中所有的行都参加编译.但是有时希望对其中一部分内容只在满足一定条件才进行编译,也就是对一部分内容指定编译的条件,这就是“条件编译”.有时,希望当满足某条件时对一组语句进行编译,而当条件不满足时则编译另一组语句. 今天看到了 #define后面只跟一个“参数” 定义宏,并在预处理过程中将其替换为空字符串(即删除).这样做主要是为了标记某些内容,使程序阅读者能够清楚标记表明的意义,同时又不影响被编译的源代码.另外,通常这些标记能被条件编译的预处理命令#ifdef.#ifndef检测

VC++ 使用预编译头

一.使用默认的预编译头       要使用预编译头,我们必须指定一个头文件,这个头文件包含我们不会经常改变的代码和其他的头文件,然后我们用这个头文件来生成一个预编译头文件(.pch文件),想必大家都知道 StdAfx.h这个文件.很多人都认为这是VC提供的一个“系统级别”的,编译器带的一个头文件.其实不是的,这个文件可以是任何名字的.我们来考察一个典型的由AppWizard生成的MFC Dialog Based 程序的预编译头文件.(因为AppWizard会为我们指定好如何使用预编译头文件,默认

C预编译, 预处理, C/C++头文件, 编译控制,

在所有的预处理指令中,#Pragma 指令可能是最复杂的了,它的作用是设定编译器的状态或者是指示编译器完成一些特定的动作.#pragma指令对每个编译器给出了一个方法,在保持与C和C++语言完全兼容的情况下,给出主机或操作系统专有的特征.依据定义,编译指示是机器或操作系统专有的,且对于每个编译器都是不同的. 其格式一般为: #Pragma Para 其中Para 为参数,下面来看一些常用的参数. (1)message 参数. Message 参数是我最喜欢的一个参数,它能够在编译信息输出窗口中输

C/C++中的预编译指令

工作中遇到的: 一个头文件中的: #pragma warning(disable:4996)#pragma warning(disable:4244)#pragma warning(disable:4267) 不理解意思,遂查? C/C++中的预编译指令 程序的编译过程可以分为预处理.编译.汇编三部分,其中预处理是首先执行的过程,预处理过程扫描程序源代码,对其进行初步的转换,产生新的源代码提供给编译器.预处理过程读入源代码之后,会检查代码里包含的预处理指令,完成诸如包含其他源文件.定义宏.根据条

iOS中的预编译指令的初步探究

看到非常好的两篇技术文,转来方便自己查看. 转自:http://www.cnblogs.com/daiweilai/p/4234336.html 开篇 我们人类创造东西的时候有个词叫做”仿生学“!人类创造什么东西都会模仿自己来创造,所以上帝没有长成树的样子而和人长得一样,科幻片里面外星人也像人一样有眼睛有鼻子……但是人类自己创造的东西如果太像自己,自己又会吓尿(恐怖谷效应),人类真是奇葩:奇葩的我们在20世纪创造了改变世界的东西——计算机(电脑),不用怀疑,这货当然也是仿生学!这货哪里长得像人了

c语言中条件编译相关的预编译指令

一. 内容概述 本文主要介绍c语言中条件编译相关的预编译指令,包括#define.#undef.#ifdef.#ifndef.#if.#elif.#else.#endif.defined. 二.条件编译 条件编译是根据实际定义宏(某类条件)进行代码静态编译的手段.可根据表达式的值或某个特定宏是否被定义来确定编译条件. 最常见的条件编译是防止重复包含头文件的宏,形式跟下面代码类似: 1 #ifndef ABCD_H 2 #define ABCD_H 3 4 // ... some declarat

iOS中的预编译指令

iOS中的预编译指令的初步探究 目录[+] 开篇 我们人类创造东西的时候有个词叫做”仿生学“!人类创造什么东西都会模仿自己来创造,所以上帝没有长成树的样子而和人长得一样,科幻片里面外星人也像人一样有眼睛有鼻子……但是人类自己创造的东西如果太像自己,自己又会吓尿(恐怖谷效应),人类真是奇葩:奇葩的我们在20世纪创造了改变世界的东西——计算机(电脑),不用怀疑,这货当然也是仿生学!这货哪里长得像人了??别不服,先听我说完,先把你的砖头放下.狭义的仿生学是外形上仿生嘛,其实广义上仿生学还可以原理的仿生