performSelector may cause a leak because its selector is unknown
转自:http://www.jianshu.com/p/6517ab655be7
问题
我在 ARC 模式下编译出了这个 warning:
"performSelector may cause a leak because its selector is unknown".
我的代码是这么写的:
[_controller performSelector:NSSelectorFromString(@"someMethod")];
为什么会有这个 warning 呢?我知道编译器无法检查实际上有没有这个 selector,不过这为什么会造成内存泄漏呢?代码应该怎么改才能消除这个 warning?
答案
答案1:单纯消除 warning
Scott Thompson,1100 票
LLVM 3.0 编译器可以用以下代码消除 warning:
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Warc-performSelector-leaks" [self.ticketTarget performSelector: self.ticketAction withObject: self];
#pragma clang diagnostic pop
如果在多个地方都要用,可以定义一个宏:
#define SuppressPerformSelectorLeakWarning(Stuff) \
do { \
_Pragma("clang diagnostic push") \
_Pragma("clang diagnostic ignored \"-Warc-performSelector-leaks\"") \
Stuff; \
_Pragma("clang diagnostic pop") \
} while (0)
用的时候:
SuppressPerformSelectorLeakWarning(
[_target performSelector:_action withObject:self]
);
如果需要返回值:
id result;
SuppressPerformSelectorLeakWarning(
result = [_target performSelector:_action withObject:self]
);
答案2:详细解释和正统解决
wbyoung,768 赞
解决方案
编译器报这个 warning 是有原因的,一般不应该直接忽略,而且消除这个 warning 并不难。如下即可:
if (!_controller) { return; }
SEL selector = NSSelectorFromString(@"someMethod");
IMP imp = [_controller methodForSelector:selector];
void (*func)(id, SEL) = (void *)imp;
func(_controller, selector);
或者写得紧密一些(不过可读性差一些,也少了类型检查):
SEL selector = NSSelectorFromString(@"someMethod");
((void (*)(id, SEL))[_controller methodForSelector:selector])(_controller, selector);
代码解释
这一堆代码在做的事情其实是,向 controller 请求那个方法对应的 C 函数指针。所有的NSObject
都能响应methodForSelector:
这个方法,不过也可以用 Objective-C runtime 里的class_getMethodImplementation
(只在 protocol 的情况下有用,id<SomeProto>
这样的)。这种函数指针叫做IMP
,就是typedef
过的函数指针(id (*IMP)(id, SEL, ...)
[1])。它跟方法签名(signature)比较像,虽然可能不是完全一样。
得到IMP
之后,还需要进行转换,转换后的函数指针包含 ARC 所需的那些细节(比如每个 OC 方法调用都有的两个隐藏参数self
和_cmd
)。这就是代码第 4 行干的事(右边的那个(void *)
只是告诉编译器,不用报类型强转的 warning)。
最后一步,调用函数指针[2]。
更复杂的例子
如果 selector 接收参数,或者有返回值,代码就需要改改:
SEL selector = NSSelectorFromString(@"processRegion:ofView:");
IMP imp = [_controller methodForSelector:selector];
CGRect (*func)(id, SEL, CGRect, UIView *) = (void *)imp;
CGRect result = _controller ?
func(_controller, selector, someRect, someView) : CGRectZero;
为什么会有这个 warning
原因是这样的:我们在 ARC 下调一个方法,runtime 需要知道对于返回值该怎么办。返回值可能有各种类型:void
,int
,char
,NSString *
,id
等等。ARC 一般是根据返回值的头文件来决定该怎么办的[3],一共有以下 4 种情况[4]:
- 直接忽略(如果是基本类型比如
void
,int
这样的)。 - 把返回值先 retain,等到用不到的时候再 release(最常见的情况)。
- 不 retain,等到用不到的时候直接 release(用于
init
、copy
这一类的方法,或者标注ns_returns_retained
的方法)。 - 什么也不做,默认返回值在返回前后是始终有效的(一直到最近的 release pool 结束为止,用于标注
ns_returns_autoreleased
的方法)。
而调performSelector:
的时候,系统会默认返回值并不是基本类型,但也不会 retain、release,也就是默认采取第 4 种做法。所以如果那个方法本来应该属于前 3 种情况,都有可能会造成内存泄漏。
对于返回void
或者基本类型的方法,就目前而言你可以忽略这个 warning,但这样做不一定安全。我看过 Clang 在处理返回值这块儿的几次迭代演进。一旦开着 ARC,编译器会觉得从performSelector:
返回的对象没理由不能 retain,不能 release。在编译器眼里,它就是个对象。所以,如果返回值是基本类型或者void
,编译器还是存在会 retain、release 它的可能,然后直接导致 crash。
带参数调用
类似地,performSelector:withObject:
也会报同一个 warning,因为不指明怎么处理参数也会有同样的问题。ARC 允许为方法参数标注consumed
,如果你调的方法有这种标注,最终可能导致把消息发给僵尸对象然后 crash。要解决这个问题可以用桥接(bridged casting),但是最好最简单的方法还是我上面写的用IMP
和函数指针的方法。不过给参数标 consumed 是比较少见的,所以这个问题也不容易发生。
静态 selector
有趣的是,下面这种静态声明的 selector 就不会出 warning:
[_controller performSelector:@selector(someMethod)];
原因是,这种情况下编译器就能在编译阶段得到关于这个 selector 的全部信息,不需要默认任何事情。
[1]: 所有的 Objective-C 方法都有两个隐藏的参数,self
和_cmd
,调用时自动加的。 ↩
[2]: 在 C 里调用NULL
方法是不安全的。而if (!_controller) { return; }
这一句保证controller
不为空,所以我们一定能从methodForSelector:
得到一个IMP
(虽然可能只是_objc_msgForward
,进入消息转发系统)。基本上,有了这行检查,就能保证我们有方法可调。 ↩
[3]: 实际上,如果返回值的类型是id
,而你又没 import 对应的头文件,它是有可能做出错误处理的。有可能会 crash 在一块编译器以为安全的代码里。这种情况很罕见,但还是有发生的可能。一般来说,如果编译器不知道该选哪个方法签名,它会报一个 warning 的。 ↩
[4]: 更多细节请参考 ARC 的文档 retain 返回值 和 不 retain 返回值。 ↩
performSelector may cause a leak because its selector is unknown的更多相关文章
- ios之"performSelector may cause a leak because its selector is unknown"警告原因及其解决办法
问题描述 项目中使用到了从字符串创建选择器,编译时发现警告:"performSelector may cause a leak because its selector is unknown ...
- PerformSelector may cause a leak because its selector is unknown 解决方法
我的技术博客经常被流氓网站恶意爬取转载.请移步原文:http://www.cnblogs.com/hamhog/p/3801030.html,享受整齐的排版.有效的链接.正确的代码缩进.更好的阅读体验 ...
- performSelector may cause a leak because its selector is unknown解决
解决方法 SEL selector = NSSelectorFromString(@"applySketchFilter:"); IMP imp = [FWApplyFilter ...
- objective-c "performSelector may cause a leak because its selector is unknown".
#define SuppressPerformSelectorLeakWarning(Stuff) \ do { \ _Pragma("clang diagnostic push" ...
- warning:performSelector may cause a leak because its selector
warning:performSelector may cause a leak because its selector 在ARC项目中使用 performSelector: withObj ...
- performSelector的原理以及用法
一.performSelector调用和直接调用区别下面两段代码都在主线程中运行,我们在看别人代码时会发现有时会直接调用,有时会利用performSelector调用,今天看到有人在问这个问题,我便做 ...
- Objective-C中一种消息处理方法performSelector: withObject:
Objective-C中调用函数的方法是“消息传递”,这个和普通的函数调用的区别是,你可以随时对一个对象传递任何消息,而不需要在编译的时候声明这些方法.所以Objective-C可以在runtime的 ...
- 【转】Objective-C中一种消息处理方法performSelector: withObject:
原文 : http://www.cnblogs.com/buro79xxd/archive/2012/04/10/2440074.html Objective-C中调用函数的方法是“消息传递”,这 ...
- iOS警告收录及科学快速的消除方法
来自: http://www.cnblogs.com/dsxniubility/p/4757760.html iOS警告收录及科学快速的消除方法 前言:现在你维护的项目有多少警告?看着几百条警 ...
随机推荐
- [转]通过Spring Boot三分钟创建Spring Web项目
来源:https://www.tianmaying.com/tutorial/project-based-on-spring-boot Spring Boot简介 接下来我们所有的Spring代码实例 ...
- 原 HTML5 requestFullScreen&exitFullscreen全屏兼容方案
摘要: html5 video全屏实现方式 首先来说,这个标题具有误导性,但这样设置改标题也是主要因为video使用的比较多 在html5中,全屏方法可以适用 ...
- python基础——函数
1.内置函数的调用: 可以在官方网站查找内置函数及说明,也可以通过help(abs)函数查看相应的信息. https://docs.python.org/3/library/functions.htm ...
- P1309 瑞士轮 排序选择 时间限制 归并排序
题目背景 在双人对决的竞技性比赛,如乒乓球.羽毛球.国际象棋中,最常见的赛制是淘汰赛和循环赛.前者的特点是比赛场数少,每场都紧张刺激,但偶然性较高.后者的特点是较为公平,偶然性较低,但比赛过程往往十分 ...
- C/C++中如何获取数组的长度(宏&模板)
C.C++中没有提供 直接获取数组长度的函数.对于存放字符串的字符数组提供了一个strlen函数获取长度,那么对于其他类型的数组如何获取他们的长度呢?其中一种方法是使 用 sizeof(array) ...
- window.open 和showModalDialog的返回值
方法: 1: 在父级页面 test.aspx 的点击<input type="button" id="btnShow" onclick="sh ...
- Nowcoder contest 392 I 逛公园 (无向图割边模板)
<题目链接> 题目描述: 月月和华华一起去逛公园了.公园很大,为了方便,可以抽象的看成一个N个点M条边的无向连通图(点是景点,边是道路).公园唯一的入口在1号点,月月和华华要从这里出发,并 ...
- angular7一周学习
ng new xxx 创建一个项目 ng serve --open 执行一个项目 angular 使用socket.io 报错 找到polyfills.ts添加 (window as any).glo ...
- ORA-01000
A ResultSet object is automatically closed by the Statement object that generated it when that State ...
- 在Eclipse中使用Junit进行单元测试
单元测试与Junit4 单元测试(unit testing),是指对软件中的最小可测试单元进行检查和验证.单元测试是在软件开发过程中要进行的最低级别的测试活动,软件的独立单元将在与程序的其他部分相隔离 ...