id & instancetype

一、 instancetype关键字

官方文档 先进的OC

Use the instancetype keyword as the return type of methods that return an instance of the class they are called on (or a subclass of that class). These methods include alloc, init, and class factory methods.
Using instancetype instead of id in appropriate places improves type safety in your Objective-C code. For example, consider the following code:

大意是:方法中含有alloc, init 或者是类的工厂方法,使用instancetype关键字代替id。提高oc代码的安全

//样例代码

@interface MyObject : NSObject
+ (instancetype)factoryMethodA;
+ (id)factoryMethodB;
@end
 
@implementation MyObject
+ (instancetype)factoryMethodA { return [[[self class] alloc] init]; }
+ (id)factoryMethodB { return [[[self class] alloc] init]; }
@end
 
void doSomething() {
    NSUInteger x, y;
   //编译报错
    x = [[MyObject factoryMethodA] count]; // Return type of +factoryMethodA is taken to be "MyObject *"
  //不报错 
    y = [[MyObject factoryMethodB] count]; // Return type of +factoryMethodB is "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

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容