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等申明一个对象时使用。