項目中使用到了從字符串創(chuàng)建選擇器什黑,編譯時發(fā)現(xiàn)警告:"performSelector may cause a leak because its selector is unknown"
(因為performSelector的選擇器未知可能會引起泄漏)鲸睛,為什么在ARC模式下會出現(xiàn)這個警告?
經過搜索后,在Stackoverflow上發(fā)現(xiàn)了一個令人滿意的答案儡首。見http://stackoverflow.com/questions/7017281/performselector-may-cause-a-leak-because-its-selector-is-unknown。
原因
在ARC模式下偏友,運行時需要知道如何處理你正在調用的方法的返回值蔬胯。這個返回值是可以是任意值,如void约谈、int笔宿、char、NSString棱诱、id等等泼橘。ARC通過頭文件的函數(shù)定義來得到這些信息。所以平時我們用到的靜態(tài)選擇器不會出現(xiàn)這個警告迈勋。因為在編譯期間炬灭,這些信息都已經確定。
如:
...
[someController performSelector:@selector(someMethod)];
...
- (void)someMethod
{
//bla bla...
}
而使用[someController performSelector: NSSelectorFromString(@"someMethod")];
時ARC并不知道該方法的返回值是什么靡菇,以及該如何處理重归?該忽略?還是標記為ns_returns_retained
還是ns_returns_autoreleased?
解決辦法
1.使用函數(shù)指針方式
SEL selector = NSSelectorFromString(@"someMethod");
IMP imp = [_contorller methodForSelector:selecotr];
CGRect (*func)(id, SEL) = (void *)imp;
CGRect result = func(_controller, selector);
當有額外參數(shù)時厦凤,如
SEL selector = NSSelectorFromString(@"processRegion:ofView:");
IMP imp = [_controller methodForSelector:selector];
CGRect (*func)(id, SEL, CGRect, UIView *) = (void *)imp;
CGRect result = func(_controller, selector,someRect,someView);
2.使用宏忽略警告
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Warc-performSelector-leaks"
[someController performSelector:NSSelectorFromString(@"someMethod")];
#pragma clang diagnostic pop
通過使用#pragma clang diagnostic push/pop
鼻吮,你可以告訴Clang編譯器僅僅為某一特定部分的代碼來忽視特定警告。
如果需要忽視的警告有多處较鼓,可以定義一個宏
#define SuppressPerformSelectorLeakWarning(Stuff) \
do { \
_Pragma("clang diagnostic push") \
_Pragma("clang diagnostic ignored \
"-Warc-performSelector-leaks\"") \
Stuff; \
_Pragma("clang diagnostic pop") \
} while (0)
在產生警告也就是performSelector的地方用使用該宏椎木,如
SuppressPerformSelectorLeakWarning( [_target performSelector:_action withObject:self]);
如果需要performSelector返回值的話,
id result;
SuppressPerformSelectorLeakWarning( result = [_target performSelector:_action withObject:self]);
3.使用afterDelay(不推薦)
[self performSelector:aSelector withObject:nil afterDelay:0.0];
如果在接受范圍內博烂,允許下一個runloop執(zhí)行香椎,可以這么做。