1、先來幾個常用的:
// 是否高清屏
#define isRetina ([UIScreen instancesRespondToSelector:@selector(currentMode)] ? CGSizeEqualToSize(CGSizeMake(640, 960), [[UIScreen mainScreen] currentMode].size) : NO)
// 是否模拟器
#define isSimulator (NSNotFound != [[[UIDevice currentDevice] model] rangeOfString:@"Simulator"].location)
// 是否iPad
#define isPad (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad)
#define someThing (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPad)? ipad: iphone
2、基本的使用:
//定義π值 3.1415926
#define PI 3.1415926
//則在程式用可以如下使用
double i=2*PI*3;
//效果相當于 double i=2*3.1415926*3;
//預處理指令可以定義任何符合格式的形式,例如判斷年份是否閏年
#define IS_LEAP_YEAR year%4==0&&year%100!=0||year%400==0
//使用時則可以直接
if(IS_LEAP_YEAR)
//或者可以定義一個參數
#define IS_LEAP_YEAR(y) y%4==0&&y%100!=0||y%400==0
//使用時則可以直接
int ys=2012;
if(IS_LEAP_YEAR(ys))
//通常預處理程式定義在一行 如果好分行 比如說太長需要換行 需要使用“/”符号 表示還有下一行,多行分列也是如此,例:
#Define IS_LEAP_YEAR year%4==0&&year%100!=0/
||year%400==0
//宏定義參數後邊放一個# 那麼在調用該宏時,預處理程式将根據宏參數建立C風格的常量字元串 例:
#define STR(x) # x
//将會使得 随後調用的
NSLOG(STR(Programming in Objective-c./n));
//顯示結果為 Programming in Objective-c./n
3、關于#與##的操作符:
<1>.宏定義中字元串化操作符#:
#的功能是将其後面的宏參數進行字元串化操作,意思就是對它所應用的宏變量通過替換後在其左右各加上一個雙引号。例如
#define WARN_IF(EXPR)\
do {\
if (EXPR)\
fprintf(stderr, "Warning: " #EXPR "\n");\
} while(0)
上面代碼中的反斜線\主要用來轉譯換行符,即屏蔽換行符。
那麼如下的代碼調用:
WARN_IF(divider == 0);
将被解析為:
if (divider == 0)\
fprintf(stderr, "Warning: " "divider == 0" "\n");\
} while(0);
注意能夠字元串化操作的必須是宏參數,不是随随便便的某個子串(token)都行的。
<2>.宏定義中的連接配接符##:
連接配接符##用來将兩個token連接配接為一個token,但它不可以位于第一個token之前or最後一個token之後。注意這裡連接配接的對象隻要是token就行,而不一定是宏參數,但是##又必須位于宏定義中才有效,因其為編譯期概念(比較繞)。
#define LINK_MULTIPLE(a, b, c, d) a##_##b##_##c##_##d
typedef struct _record_type LINK_MULTIPLE(name, company, position, salary);
/*
* 上面的代碼将被替換為
* typedef struct _record_type name_company_position_salary;
*/
又如下面的例子:
#define PARSER(N) printf("token" #N " = %d\n", token##N)
int token64 = 64;
如下調用宏:
PARSER(64);
printf("token" "64" " = %d\n", token64);
在obj-c中,如果我有如下定義:
#define _X(A, B) (A#B)
#define _XX(A, B) _X([NSString stringWithFormat:@"%@_c", A], B)
gcc将報錯!
正确的寫法為:
#define _XX(A, B) _X(([NSString stringWithFormat:@"%@_c", A]), B)
4、再來個宏定義 object-c 單例
#define GTMOBJECT_SINGLETON_BOILERPLATE(_object_name_, _shared_obj_name_)
static _object_name_ *z##_shared_obj_name_ = nil;
+ (_object_name_ *)_shared_obj_name_ {
@synchronized(self) {
if (z##_shared_obj_name_ == nil) {
/* Note that ‘self’ may not be the same as _object_name_ */
/* first assignment done in allocWithZone but we must reassign in case init fails */
z##_shared_obj_name_ = [[self alloc] init];
_GTMDevAssert((z##_shared_obj_name_ != nil), @”didn’t catch singleton allocation”);
}
}
return z##_shared_obj_name_;
}
+ (id)allocWithZone:(NSZone *)zone {
z##_shared_obj_name_ = [super allocWithZone:zone];
return z##_shared_obj_name_;
/* We can’t return the shared instance, because it’s been init’d */
_GTMDevAssert(NO, @”use the singleton API, not alloc+init”);
return nil;
- (id)retain {
return self;
- (NSUInteger)retainCount {
return NSUIntegerMax;
- (void)release {
- (id)autorelease {
- (id)copyWithZone:(NSZone *)zone {
}
5、條件編譯:
#if !defined(FCDebug) || FCDebug == 0
#define FCLOG(...) do {} while (0)
#define FCLOGINFO(...) do {} while (0)
#define FCLOGERROR(...) do {} while (0)
#elif FCDebug == 1
#define FCLOG(...) NSLog(__VA_ARGS__)
#define FCLOGERROR(...) NSLog(__VA_ARGS__)
#elif FCDebug > 1
#define FCLOGINFO(...) NSLog(__VA_ARGS__)
#endif
6、參照C語言的預處理指令簡介 :
#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 顯示編譯錯誤資訊
如何聯系我:【萬裡虎】www.bravetiger.cn
【QQ】3396726884 (咨詢問題100元起,幫助解決問題500元起)
【部落格】http://www.cnblogs.com/kenshinobiy/