__strong 限定符用于非 ARC 项目

use*_*931 3 objective-c ios automatic-ref-counting

该项目是非 ARC 启用的,但是我们(错误地)使用了符合 ARC 的代码库 - 特别是一个创建单例对象的代码库,如 GCDSingleton.h 中定义的那样:

#define DEFINE_SHARED_INSTANCE
+ (id)sharedInstance
{
  static dispatch_once_t pred = 0;
  __strong static id _sharedObject = nil;
  dispatch_once(&pred, ^{
    _sharedObject = ^{return [[self alloc] init];}();
  });
  return _sharedObject;
}
Run Code Online (Sandbox Code Playgroud)

即使共享对象是用 __strong 限定符定义的,这似乎也有效。我想知道为什么这不会导致错误或至少是警告(最新的 Xcode 4.6 和 ios 6 sdk)。此外,由于该项目未启用 ARC,那么 __strong 限定符究竟在做什么(如果有的话)?

Gab*_*lla 5

在 MRC 代码中,__strong被简单地忽略。

我试图编译一个简单的例子

#import <Foundation/Foundation.h>

int main(int argc, char const *argv[]) {
    __strong NSString * foo = [[NSString alloc] initWithFormat:@"Hello, %s", argv[1]];
    NSLog(@"%@", foo);
}
Run Code Online (Sandbox Code Playgroud)

带弧光

clang -fobjc-arc test.m -S -emit-llvm -o arc.ir
Run Code Online (Sandbox Code Playgroud)

并且没有ARC

clang -fno-objc-arc test.m -S -emit-llvm -o mrc.ir
Run Code Online (Sandbox Code Playgroud)

并区分 llvm IR 输出。

这是结果 diff mrc.ir arc.ir

54a55,56
>   %17 = bitcast %0** %foo to i8**
>   call void @objc_storeStrong(i8** %17, i8* null) nounwind
63a66,67
> declare void @objc_storeStrong(i8**, i8*)
> 
Run Code Online (Sandbox Code Playgroud)

所以基本上ARC和MRC之间的唯一区别就是增加了一个objc_storeStrong调用。


顺便说一句,没有__strong限定符的相同代码将产生完全相同的结果,因为它__strong是 ARC 中变量的默认限定符。