instancetype意思为实例化,instancetype与和id一样,都可以指向一个继承了Object(或者NSObject)类的对象。
区别在于:instancetype只能作为方法返回值,会返回该方法所在的类的实例化对象,所以instancetype被称为关联返回类型。
使用instancetype会在编译时进行类型检查,有利于开发者在编译阶段发现错误。
根据 Cocoa 的命名规则,满足下述命名规则的方法应该返回该方法所在类的实例化对象,这些方法被称为是关联返回类型的方法:
以 alloc 或 new 开头的类方法以 autorelease、init、retain 或 self 开头的实例方法。例如:
+ (instancetype)alloc; - (instancetype)init; //因此这里alloc和init返回的类型都是NSArray* NSArray *arr = [[NSArray alloc]init];我们先创建一个Order类,添加两个自定义的alloc和init方法,然后再创建一个Order的子类OrderSubclass,OrderSubclass什么方法都不用添加。
@interface Order : UIView +(instancetype) allocMethod; -(instancetype) initMethod; @end @implementation Order +(instancetype) allocMethod { return [[Order alloc]init]; //<-这里是重点 } -(instancetype) initMethod { return [[Order alloc]init]; //<-这里是重点 } @end然后使用我们自定义的alloc和init方法,分别实例化Order和OrderSubclass,看一下有什么问题。
Order * order = [[Order allocMethod]initMethod]; OrderSubclass *orderSub = [[OrderSubclass allocMethod]initMethod]; NSLog(@"order: %@",[order class]); NSLog(@"orderSub:%@",[orderSub class]); log输出: order: Order orderSub:Order从log输出来看,OrderSubclass的类型居然是是Order,这显然不是我们所希望的。
原因就出在前面标记的重点处,如果子类使用了父类的自定义实例化方法,那么就会得到父类的类型,而不是子类自己的类型。
所以我们应该使用[[self class] alloc]来进行实例化,而不是[类名 alloc]。
@implementation Order +(instancetype) allocMethod { return [[[self class] alloc]init]; //<-这里是重点 } -(instancetype) initMethod { return [[[self class] alloc]init]; //<-这里是重点 } @end log输出: order: Order orderSub:OrderSubclass