iOS笔记-关于instancetype

参考Adopting Modern Objective-C

instancetype说明

instancetype是clang 3.5开始提供的一个关键字,跟id类似,用于表示某个方法返回的未知类型的Objective-C对象。

使用instancetype的好处

考虑一下代码:

@interface MyObject : NSObject
+ (instancetype)factoryMethodA;
+ (id)factoryMethodB;
@end
 + (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"
}

因为instancetype具有对象类型检测功能,因此factoryMethodA函数的返回对象类型就是MyObject,而MyObject是没有count这个函数的,所以编译器会在x行这里提示警告"’MyObject’ may not respond to ‘count’"

而id类型不具备类型检测功能,其泛指所有class,而count函数有可能存在任何一个class中(比如NSArray),所以y对应的代码不会警告,不过在真正运行到此处代码时,程序就会crash

使用注意

  • instancetype只能作为函数返回值,不能用来定义定义变量。
// 错误写法  
- (void)setValue:(instancetype)value
{
} 
// 正确写法
- (void)setValue:(id)value
{
}
  • 用instancetype作为函数返回值返回的是函数所在类型的类型。
/*
这种写法编译器会报警告:Incompatible pointer types initializing ’NSString *’ 
*/
NSString *string = [MyObject factoryMethodA];

// 正确写法
MyObject *myObject = [MyObject factoryMethodA];

补充说明

为了在继承中完美使用工程方法,工程方法的实现最好使用[self class]而不要使用[类名 alloc]的方式。

// 这个方法返回的对象是对应子类的对象
+ (instancetype)factoryMethodA
{
    return [[[self class] alloc] init];
}
// 这个方法返回的对象是MyObject对应的对象,而不是子类的对象
+ (instancetype)factoryMethodB
{
    return [[MyObject alloc] init];
}
// 创建一个MyObject的子类MyObjectSubclass
NSLog(@"%@",[[MyObjectSubclass factoryMethodA] class]);
NSLog(@"%@",[[MyObjectSubclass factoryMethodB] class]);

// 输出日租
MyObjectSubclass
MyObject

综上所述

instancetype是函数返回值类型,用来获取函数所在方法的class对象,比如用作类方法、alloc、init、new等申明一个对象时使用。

你可能感兴趣的:(iOS笔记-关于instancetype)