一些宏的应用,包括NSLog
創建一個UIViewController,免得類名與資源名寫錯
#define CREATCONTROLLER(ClassName)? ? [[ClassName alloc] initWithNibName:[NSString stringWithFormat:@"%s",#ClassName] bundle:nil];
?
創建一個錯誤處理實體
?
#define ERROR_INFO(ErrorDomain,ErrorCode,ErroeDescript) [[NSError alloc] initWithDomain:[NSString stringWithFormat:ErrorDomain] code:ErrorCode userInfo:[NSDictionary dictionaryWithObject:ErroeDescript forKey:NSLocalizedDescriptionKey]];
#define ERROR_DICTINFO(ErrorDomain,ErrorCode,ErroeDictionary) [[NSError alloc] initWithDomain:[NSString stringWithFormat:ErrorDomain] code:ErrorCode userInfo:ErroeDictionary];
#define ERROR_NOINFO(ErrorDomain,ErrorCode) [[NSError alloc] initWithDomain:[NSString stringWithFormat:ErrorDomain] code:ErrorCode userInfo:nil];
?
?
針對debug的日志打印
#ifdef DEBUG
//#define CLog(format, ...) NSLog(format, ##__VA_ARGS__);
#define CLog(fmt, ...) NSLog((@"%s [Line %d] " fmt), __PRETTY_FUNCTION__, __LINE__, ##__VA_ARGS__);
#define DEBUGPrintf(fmt, ...)? printf(fmt,##__VA_ARGS__);
// ALog always displays output regardless of the DEBUG setting ?
#define ALog(fmt, ...) NSLog((@"%s [Line %d] " fmt), __PRETTY_FUNCTION__, __LINE__, ##__VA_ARGS__); ?
#else
#define CLog(format, ...)
#define DEBUGPrintf(fmt, ...)
#define ALog(fmt, ...)
#endif
?
針對模擬器的日志打印
#ifdef __i386__
#??? define NSLog(...) NSLog(__VA_ARGS__)
#else
#??? define NSLog(...) {}
#endif
上例是用C99標準中的可變參數宏來實現的。
在C/C++語言中,"…"表示可以接受不定數量的參數,理論上來講,可以是0或0以上的n個參數。
?
用可變參數宏(variadic macros)傳遞可變參數表
你可能很熟悉在函數中使用可變參數表,如:
void printf(const char* format, …);
?
直到最近,可變參數表還是只能應用在真正的函數中,不能使用在宏中。
?
C99編譯器標準終于改變了這種局面,它允許你可以定義可變參數宏(variadic macros),這樣你就可以使用擁有可以變化的參數表的宏。可變參數宏就像下面這個樣子:
?
#define debug(…) printf(__VA_ARGS__)
?
缺省號代表一個可以變化的參數表。使用保留名 __VA_ARGS__ 把參數傳遞給宏。當宏的調用展開時,實際的參數就傳遞給?printf()了。例如:
?
Debug(“Y = %d\n”, y);
?
而處理器會把宏的調用替換成:
?
printf(“Y = %d\n”, y);
?
因為debug()是一個可變參數宏,你能在每一次調用中傳遞不同數目的參數:
?
debug(“test”);?//一個參數
?
可變參數宏不被ANSI/ISO C++ 所正式支持。因此,你應當檢查你的編譯器,看它是否支持這項技術。
?
?
?
用GCC和C99的可變參數宏, 更方便地打印調試信息
gcc的預處理提供的可變參數宏定義真是好用:?
#define?dbgprint(format,args...)?\
fprintf(stderr,?format,?##args)
#else
#define?dbgprint(format,args...)
#endif
如此定義之后,代碼中就可以用dbgprint了,例如dbgprint("aaa?%s",?__FILE__);。感覺這個功能比較Cool??:em11:?
下面是C99的方法:?
???????????? printf(fmt,__VA_ARGS__)
新的C99規范支持了可變參數的宏
具體使用如下:
?以下內容為程序代碼:
?#include <stdarg.h>?#include <stdio.h>
?#define LOGSTRINGS(fm, ...) printf(fm,__VA_ARGS__)
?int main()?{????? LOGSTRINGS("hello, %d ", 10);????? return 0;?}?
?但現在似乎只有gcc才支持。
?
可變參數的宏里的‘##’操作說明
帶有可變參數的宏(Macros with a Variable Number of Arguments)
在1999年版本的ISO C?標準中,宏可以象函數一樣,定義時可以帶有可變參數。宏的語法和函數的語法類似。下面有個例子:
?
#define debug(format, ...) fprintf (stderr, format, __VA_ARGS__)
?
這里,‘…’指可變參數。這類宏在被調用時,它(這里指‘…’)被表示成零個或多個符號,包括里面的逗號,一直到到右括弧結束為止。當被調用時,在宏體(macro body)中,那些符號序列集合將代替里面的__VA_ARGS__標識符。更多的信息可以參考CPP手冊。
?
GCC始終支持復雜的宏,它使用一種不同的語法從而可以使你可以給可變參數一個名字,如同其它參數一樣。例如下面的例子:
?
#define debug(format, args...) fprintf (stderr, format, args)
?
這和上面舉的那個ISO C定義的宏例子是完全一樣的,但是這么寫可讀性更強并且更容易進行描述。
?
GNU CPP還有兩種更復雜的宏擴展,支持上面兩種格式的定義格式。
?
在標準C里,你不能省略可變參數,但是你卻可以給它傳遞一個空的參數。例如,下面的宏調用在ISO C里是非法的,因為字符串后面沒有逗號:
?
debug ("A message")
?
GNU CPP在這種情況下可以讓你完全的忽略可變參數。在上面的例子中,編譯器仍然會有問題(complain),因為宏展開后,里面的字符串后面會有個多余的逗號。
?
為了解決這個問題,CPP使用一個特殊的‘##’操作。書寫格式為:
?
#define debug(format, ...) fprintf (stderr, format, ## __VA_ARGS__)
?
這里,如果可變參數被忽略或為空,‘##’操作將使預處理器(preprocessor)去除掉它前面的那個逗號。如果你在宏調用時,確實提供了一些可變參數,GNU CPP也會工作正常,它會把這些可變參數放到逗號的后面。象其它的pasted macro參數一樣,這些參數不是宏的擴展。
怎樣寫參數個數可變的宏
一種流行的技巧是用一個單獨的用括弧括起來的的?``參數"?定義和調用宏, 參數在 宏擴展的時候成為類似?printf()?那樣的函數的整個參數列表。
if(n != 0) DEBUG(("n is %d\n", n));
?
明顯的缺陷是調用者必須記住使用一對額外的括弧。
?
gcc?有一個擴展可以讓函數式的宏接受可變個數的參數。 但這不是標準。另一種 可能的解決方案是根據參數個數使用多個宏?(DEBUG1, DEBUG2, 等等), 或者用 逗號玩個這樣的花招:
?
#define DEBUG(args) (printf("DEBUG: "), printf(args))#define _ ,
DEBUG("i = %d" _ i);
?
C99?引入了對參數個數可變的函數式宏的正式支持。在宏?``原型"?的末尾加上符號?... (就像在參數可變的函數定義中), 宏定義中的偽宏?__VA_ARGS__?就會在調用是 替換成可變參數。
最后, 你總是可以使用真實的函數, 接受明確定義的可變參數
如果你需要替換宏, 使用一個 函數和一個非函數式宏, 如?#define printf myprintf。
?
摘抄:http://www.cnblogs.com/wwwkhd/archive/2011/08/01.html
轉載于:https://www.cnblogs.com/goodleixiao/archive/2012/06/04/2534424.html
總結
以上是生活随笔為你收集整理的一些宏的应用,包括NSLog的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 644 - Immediate Deco
- 下一篇: How to Increase the