id 和instancetype的区别

一、概述

instancetype是clang 3.5开始提供的一个关键字,与id一样表示某个方法返回的未知类型的Objective-C对象。但是instancetype和id区别到底有什么区别呢?


二、关联返回类型和非关联返回类型

要搞懂id 和instancetype的异同 首先要弄懂iOS中两个概念:关联返回类型和非关联返回类型。


1、关联返回类型

根据Cocoa的命名规则,满足下述规则的方法:

(1)类方法中,以alloc或new开头

(2)实例方法中,以autorelease,init,retain或self开头

会返回一个方法所在类类型的对象,这些方法就被称为是关联返回类型的方法。换句话说,这些方法的返回结果以方法所在的类为类型。


如下:

@interface NSObject 

+ (id)alloc; 

- (id)init; 

@end

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

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

按照Cocoa的命名规则,[NSArray alloc]与[[NSArray alloc]init]返回的都为NSArray的对象。


2、非关联返回类型

@interface NSArray 

+ (id)constructAnArray; 

@end

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

[NSArray constructAnArray];

根据Cocoa的方法命名规范,得到的返回类型就和方法声明的返回类型一样,是id。

但是如果使用instancetype作为返回类型,如下:

@interface NSArray 

+ (instancetype)constructAnArray; 

@end

当使用相同方式初始化NSArray时:

[NSArray constructAnArray];

得到的返回类型和方法所在类的类型相同,是NSArray*!

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



三、instancetype和id区别


(1)id在编译的时候不能判断对象的真实类型

instancetype在编译的时候可以判断对象的真实类型


(2)如果init方法的返回值是instancetype,那么将返回值赋值给一个其它的对象会报一个警告

如果是在以前, init的返回值是id,那么将init返回的对象地址赋值给其它对象是不会报错的


(3)id可以用来定义变量, 可以作为返回值, 可以作为形参

instancetype只能用于作为返回值,例如:

//err,expected a type 

- (void)setValue:(instancetype)value 

    //do something 

就是错的,应该写成:

- (void)setValue:(id)value 

    //do something 

}


注意:以后但凡自定义构造方法, 返回值尽量使用instancetype, 不要使用id

原文链接:https://link.juejin.im/?target=http%3A%2F%2Fwww.imlifengfeng.com%2Fblog%2F%3Fp%3D485

你可能感兴趣的:(id 和instancetype的区别)