IOS-id和instancetype

在IOS中数据类型有以下几种

  1. 基本数据类型:int、float、double和char类型。
  2. 对象数据类型:就是类或协议所声明的指针类型,如NSArray * array
  3. id类型:指向OC对象的指针,可以映射任何对象指针类型指向它,或者映射它指向其他的对象。当然,也可以传递任何消息给id,但如果该id不支持这个消息就会返回一个运行时异常。

以上三种类型可以用作传递的参数类型或者返回的参数类型。

  1. -(int)aaa:(int)num
  2. -(NSArray *)bbb:(NSArray *)array
  3. -(id)ccc:(id)num

下面介绍一下只使用在返回值类型中的两种特殊类型

  1. void:无返回值类型
  2. instancetype:返回类自身的实例。

void类型好理解,只要是程序员都知道它,在这里就不更多的描述了。

    instancetype用来表示Related Result Types(关联返回类型),对于简易构造函数(convenience constructor),应该总是用instancetype。苹果在iOS 8中全面使用instancetype代替id。

    什么是关联返回类型呢?根据Cocoa的命名惯例,init, alloc这类的方法,如果以id作为返回类型,会返回类本身的类型。

请看下面的例子


@interface NSObject  
+ (id)alloc;  
- (id)init;  
@end  

@interface NSArray : NSObject  
@end  

当我们使用如下方式初始化NSArray时:


NSArray *array = [[NSArray alloc] init];  

    按照Cocoa的命名规则,语句[NSArray alloc]的类型就是NSArray,因为alloc的返回类型属于关联返回类型。同样,[[NSArray alloc]init] 的返回结果也是NSArray。

    instancetype的作用,就是使那些非关联返回类型的方法返回所在类的类型!


@interface Foo:NSObject
 - (id)initWithBar:(NSInteger)bar; // initializer
 + (instancetype)fooWithBar:(NSInteger)bar;  // convenience constructor
@end

上面代码中(id)initWithBar方法会被编译器自动编译成(instancetype)initWithBar。

使用instancetype的好处,明确该方法的返回类型,编译阶段发现错误。


[[[NSArray alloc] init] mediaPlaybackAllowsAirPlay]; //  "No visible @interface for `NSArray` declares the selector `mediaPlaybackAllowsAirPlay`"  

[[NSArray array] mediaPlaybackAllowsAirPlay]; // (No error)  

    上例中第一行代码,由于[[NSArray alloc]init]的结果是NSArray*,这样编译器就能够根据返回的数据类型检测出NSArray是否实现mediaPlaybackAllowsAirPlay方法。有利于开发者在编译阶段发现错误。

    第二行代码,由于array不属于关联返回类型方法,[NSArray array]返回的是id类型,编译器不知道id类型的对象是否实现了mediaPlaybackAllowsAirPlay方法,也就不能够替开发者及时发现错误。

[参考:] http://nshipster.cn/instancetype/