局部变量自己主动俘获

偶然在调试中发现,performSelector 方法具有自己主动俘获变量的特性。试看例如以下代码:

  1. CGFloat c = _addViewShowing ? 0 : 80;
  2. if([self respondsToSelector:@selector(jsq_setToolbarBottomLayoutGuideConstant:)]){
  3. [self performSelector:@selector(jsq_setToolbarBottomLayoutGuideConstant:) withObject:nil];
  4. ...
  5. }

这里请注意 [self performSelector:@selector(jsq_setToolbarBottomLayoutGuideConstant:) withObject:nil]; 一句。

在调用 performSelector 方法时,会自己主动把变量 CGFloat c 俘获到 jsq_setToolbarBottomLayoutGuideConstant: 方法调用中去。也就是说,相当于向该方法传递了參数 c。

值得注意的是。变量 c 的类型必须和 jsq_setToolbarBottomLayoutGuideConstant: 方法參数的类型同样。否则不会自己主动俘获。比如, c 变量为 CGFloat,而方法jsq_setToolbarBottomLayoutGuideConstant: 的參数同样也为 CGFloat:

  1. - (void)jsq_setToolbarBottomLayoutGuideConstant:(CGFloat)constant

假设你将 c 的类型改成 int。则 c 不会被自己主动俘获。

利用这个特性,我们能够在 performSelector 调用时,自己主动传递变量给目标方法,而不用通过 withObject 来传參。

自己主动俘获方法參数

假设我们将上述代码定义为一个方法:

  1. -(void)setToolbarSpaceToBottom:(CGFloat)constant{
  2. // 调用私有方法 jsq_setToolbarBottomLayoutGuideConstant
  3. CGFloat c = _addViewShowing ? 0 : 80;
  4. if([self respondsToSelector:@selector(jsq_setToolbarBottomLayoutGuideConstant:)]){
  5. [self performSelector:@selector(jsq_setToolbarBottomLayoutGuideConstant:) withObject:nil];
  6. }
  7. }

则变量 c 能够省略,由于 performSelector 会自己主动俘获方法參数 constant。将之传递给 jsq_setToolbarBottomLayoutGuideConstant: 调用。于是这种方法能够写成:

  1. -(void)setToolbarSpaceToBottom:(CGFloat)constant{
  2. if([self respondsToSelector:@selector(jsq_setToolbarBottomLayoutGuideConstant:)]){
  3. [self performSelector:@selector(jsq_setToolbarBottomLayoutGuideConstant:) withObject:nil];
  4. }
  5. }

自己主动俘获的代价

上述代码同一时候会带来一个负面作用。即在两个 @selector 引用的地方出现两个同样编译警告:

Undeclared selector ‘jsq_setToolbarBottomLayoutGuideConstant:’

这是由于 jsq_setToolbarBottomLayoutGuideConstant: 方法来自于父类。它是私有的(没有将方法进行静态声明——即未在头文件里声明)。

对一切未静态声明的方法进行 performSelector 时。编译器都会提示 Undeclared selector。

你能够用以下的技术消除它们。但这会导致自己主动俘获失效。

不能使用自己主动俘获的情况

要消灭编译警告,我们能够使用 NSSelectorFromString 来引用 selector。

比如:

  1. CGFloat c = constant;
  2. SEL sel=NSSelectorFromString(@"jsq_setToolbarBottomLayoutGuideConstant:");
  3. if([self respondsToSelector:sel]){
  4. // 忽略编译器警告
  5. #pragma clang diagnostic push
  6. #pragma clang diagnostic ignored "-Warc-performSelector-leaks"
  7. [self performSelector:sel withObject:nil];
  8. #pragma clang diagnostic pop
  9. }

但这样的情况下。变量 c 不会被自己主动俘获。假设你在 jsq_setToolbarBottomLayoutGuideConstant: 方法的第一行代码加上断点执行程序。当程序执行到断点处时,打印參数的值,你会发现其值为 NaN 。假设继续执行代码。App 会崩溃。

这样的情况下。我们无法使用自己主动俘获,因此仅仅能使用 withObject 来传递參数了。

但由于 CGFloat 不是 NSObject,无法用 [performSelector: withObjectd:] 来传參。因此要使用 NSInvocation 来调用:

  1. -(void)setToolbarSpaceToBottom:(CGFloat)constant{
  2. SEL sel=NSSelectorFromString(@"jsq_setToolbarBottomLayoutGuideConstant:");
  3. NSInvocation *invoc = [NSInvocation invocationWithMethodSignature:[[self class] instanceMethodSignatureForSelector:sel]];
  4. [invoc setSelector:sel];
  5. [invoc setTarget:self];
  6. [invoc setArgument:&constant atIndex:2];//"Indices 0 and 1 indicate the hidden arguments self and _cmd"
  7. [invoc performSelector:@selector(invoke) withObject:nil];
  8. }

performSelector 方法的自己主动俘获特性的更多相关文章

  1. 如何在方法上贴上attribute(特性)捕捉方法的异常,来实现我们的需求

    在方法上贴上attribute(特性)捕捉方法的异常,其实这么做也是为了在项目中不会大量使用try-cacth这样的语句,同时使我们的代码看起来更简洁,更直观,将逻辑业务分离使得后期维护方便.这里我们 ...

  2. vue 父组件主动获取子组件的数据和方法 子组件主动获取父组件的数据和方法

    Header.vue <template> <div> <h2>我是头部组件</h2> <button @click="getParen ...

  3. C#提高--------------获取方法返回值的自定义特性(Attribute)

    .NET(C#):获取方法返回值的自定义特性(Attribute) 转载 2013年05月08日 10:54:42 1456 来自:http://www.cnblogs.com/mgen/archiv ...

  4. Swift里performSelector方法的替代

    最近在回答StackOverflow的问题时,发现performSelector方法在Swift被去掉,Apple的注释是这个方法被去掉是因为不安全: NOTE The performSelector ...

  5. [Xcode 实际操作]八、网络与多线程-(19)使用RunLoop使PerformSelector方法延迟动作的执行

    目录:[Swift]Xcode实际操作 本文将演示使用RunLoop使PerformSelector方法延迟动作的执行. 在项目导航区,打开视图控制器的代码文件[ViewController.swif ...

  6. C# 通过反射获取方法/类上的自定义特性

    1.所有自定义属性都必须继承System.Attribute 2.自定义属性的类名称必须为 XXXXAttribute 即是已Attribute结尾 自定义属性QuickWebApi [Attribu ...

  7. 猫猫学iOS之小知识之_xcode插件的删除方法_自己主动提示图片插件KSImageNamed有时不灵_分类或宏之类不能自己主动提示,

    猫猫分享,必须精品 原创文章.欢迎转载. 转载请注明:翟乃玉的博客 地址:http://blog.csdn.net/u013357243 一:解决解决自己主动提示图片插件KSImageNamed有时不 ...

  8. [译]java9新特性:在接口中用pirvate方法让default(java8接口特性)更简练

    Java8 带来了许多改变,其中之一就是default修饰的接口方法. 这些方法改变了我们已知的接口,现在我们能够在接口中定义默认实现方法.默认实现方法的不同之处在于,在接口中用default修饰抽象 ...

  9. 【.net 深呼吸】自定义特性(Attribute)的实现与检索方法

    在.net的各个语言中,尤其是VB.NET和C#,都有特性这一东东,具体的概念,大家可以网上查,这里老周说一个非标准的概念——特性者,就是对象的附加数据.对象自然可以是类型.类型成员,以及程序集. 说 ...

随机推荐

  1. HDU 2639 Bone Collector II【01背包 + 第K大价值】

    The title of this problem is familiar,isn't it?yeah,if you had took part in the "Rookie Cup&quo ...

  2. STL+Floyd【p1690】贪婪的Copy

    Description Copy从卢牛那里听说在一片叫yz的神的领域埋藏着不少宝藏,于是Copy来到了这个被划分为个区域的神地.卢牛告诉了Copy这里共有个宝藏,分别放在第Pi个(1<=Pi&l ...

  3. [P2023][AHOI2009]维护序列(线段树)

    题目描述 老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成. 有长为N的数列,不妨设为a1,a2,…,aN .有如下三种操作形式: (1)把数列中的一段数全部乘一个值; (2)把数列中的一 ...

  4. ARC 098 D - Xor Sum 2

    Problem Statement There is an integer sequence A of length N. Find the number of the pairs of intege ...

  5. unity3d 网页游戏客户端工程构建方案

    将一个项目分为两个编辑环境,一个是editor,一个是target. editor只是策划人员拖拖拽拽编辑场景,打包时程序自动将每个场景资源打包生成一个XXX.unity3d文件,并最后生成一个场景配 ...

  6. @selector和SEL

    遇到selector发现不是很明白,网上搜到的零零星星的介绍也不成体系,索性自己翻译一下,加深一下印象.原文来自官方API文档下的Selectors. Selectors 在OC中,selector有 ...

  7. IE8下 input标签内padding失效

    在做网页兼容时 发现在ie8下的input内用padding失效 为了达到居中文字的效果 使用line-height可以解决问题

  8. jquery 纯JS设置select下拉框,并默认选中第一个

    //html页面<select id="payWay" class="easyui-combobox" name="payWay" s ...

  9. gcc 4.8更新gcc 4.9 5.4版本等

    转载:http://www.linuxidc.com/Linux/2017-01/139976.htm 如果还在使用较旧版本的Ubuntu,或者是Ubuntu LTS,那么我们是很难体验新版gcc的. ...

  10. Java 学习之网络编程案例

    网络编程案例 一,概念 1,网络编程不等于网站编程 2,编程只和传输层打交道,即TCP和UDP两个协议 二,案例 1,TCP实现点对点的聊天 Server端:两个输入流:读客户端和控制台,一个输出端: ...