1

内容提要

通过阅读本篇文章,希望您能了解:如何在 ARC 下,强制某个变量调用指定的方法?

前言

首先让我们先感谢以 Blaine Garst 和 Patrick Beard 为代表的开发者。他们为 Objective-C 这门语言添加了众多特性,而其中最令人喜爱的莫过于 ARC。

相信所有和我一样经历过手动管理内存时代的人,都会对这个特性感到非常的喜爱。

然而,有所得必有所失。本文将会讲讲 ARC 带来的一些“麻烦”。

一道简单的面试题

在 MRR 环境下,下面的代码会?Compile Error / Runtime Crash / NSLog…?

//NSObject+Sun.h
 @interface NSObject (Sun)
 - (void)shining;
 @end

//NSObject+Sun.m
 @implementation NSObject (Sun)
 - (void)shining
 {
    NSLog(@"shining");
 }

 @end
 
//main.m
#import <Foundation/Foundation.h>

 int main(int argc, const char * argv[]) {
  @autoreleasepool {
      [NSObject shining];
}
return 0;
}

答案很简单,虽然会产生一个警告⚠️ Class method '+shining' not found (return type defaults to 'id'),但是可以正常输出 shining

对此感到困惑的读者可以查看拙作 http://www.jianshu.com/p/e309...

具体到此问题,可以查看 @halfrost 的文章 神经病院Objective-C Runtime住院第二天—消息发送与转发

升级版面试题

刚才的代码在 ARC 环境下,也能正常输出 shining 吗?
答案是,不会。它会产生一个编译错误❗️。错误的定义如下所示。

def err_arc_may_not_respond : Error<
  "no visible @interface for %0 declares the selector %1">;

解决方案也很简单。在 main.m 文件中添加一行代码即可。

#import "NSObject+Sun.h"

问题分析

为什么在 MRR 下面能够正常运行的代码,却无法在 ARC 下编译呢?

实际上,在默认情况下,在 ARC 下面进行编译时,编译器需要知道该方法是如何声明的,只有这样,编译器才能自动插入相关的内存管理代码并进行代码优化。

那么,是否可以强制某个变量执行指定的方法呢?答案是:有。
可以通过下面主动声明的方式,告诉编译器,请执行该方法。

[[NSObject new] performSelector:@selector(shining)];
[NSObject performSelector:@selector(shining)];

下期预告

相信很多人都没有看明白,为什么在默认情况下,编译器需要知道方法是如何声明的?

如果你知道原因,欢迎在下方进行评论。

如果你感到好奇,敬请期待下期。


酷酷的哀殿
1.1k 声望21 粉丝

百度 RD