需求:html
有10个类, 每一个类里头都有n个方法(前提是方法名有规律可循,好比 setA0,setA1…) 若是挨个去init类, 而后挨个调用方法,这样你一天就不用干别的了ios
简单的解决方法能够参考这个: iPhone开发中,动态调用类和方法git
虽说performSelector有返回值,可是在ARC下会有警告,由于方法名是动态的,系统并不知道返回值是什么样的类型。解决方法能够见这篇文章: "performSelector may cause a leak because its selector is unknown"警告缘由及其解决办法github
全文转载过来:函数
项目中使用到了从字符串建立选择器,编译时发现警告:"performSelector may cause a leak because its selector is unknown"(由于performSelector的选择器未知可能会引发泄漏),为何在ARC模式下会出现这个警告?oop
通过搜索后,在Stackoverflow上发现了一个使人满意的答案。见http://stackoverflow.com/questions/7017281/performselector-may-cause-a-leak-because-its-selector-is-unknown。spa
在ARC模式下,运行时须要知道如何处理你正在调用的方法的返回值。这个返回值能够是任意值,如void
,int
,char
,NSString
,id
等等。ARC经过头文件的函数定义来获得这些信息。因此平时咱们用到的静态选择器就不会出现这个警告。由于在编译期间,这些信息都已经肯定。指针
如:code
... [someController performSelector:@selector(someMethod)]; ... - (void)someMethod { //bla bla...}
而使用[someController performSelector: NSSelectorFromString(@"someMethod")];
时ARC并不知道该方法的返回值是什么,以及该如何处理?该忽略?仍是标记为ns_returns_retained
仍是ns_returns_autoreleased
?orm
SEL selector = NSSelectorFromString(@"someMethod");IMP imp = [_controller methodForSelector:selector];void (*func)(id, SEL) = (void *)imp;func(_controller, selector);
当有额外参数时,如
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);
#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] );
[self performSelector:aSelector withObject:nil afterDelay:0.0];
若是在接受范围内,容许在下一个runloop执行,能够这么作。xCode5没问题,但据反映,xCode6的话这个不能消除警告。