android工程师开发IOS oc浅析(11)之instancetype与id的区别

id类型及应用场景
1、用NSObject访问子类对象方法
//需要强制转换,是静态类型
NSObject*obj = cat;
[(Cat*)objrun];
2、id类型
id是动态类型,程序只有在运行时才能确定它的具体类型。是指向哪个对象的。
id是一种通用的对象类型,它可以用来存储属于任何类的对象也可以理解为万能指针
注意 在id的定义中,已经包好了*号。id指针只能指向os的对象。
id类型的定义
Typedefstruct objcobject{
Class isa;
} *id;
局限性:调用一个不存在的方法,编译器会马上报错。
iOS5之后推出了instancetype类型
instancetype和id的异同
1)相同点都可以作为方法的返回类型
2)不同点
1,instancetype可以返回和方法所在类相同类型的对象,id只能返回未知类型的对象;
2,instancetype只能作为返回值,不能像id那样作为参数,
3、id类型应用场景
//动物类
Animal *animal = [Animal new]; //狗的类,狗继承自动物
Dog *dog = [Dog new]; //猫的类,猫继承自动物
Cat *cat = [Cat new]; //定义id类型
id obj = cat; //id obj = dog;
[objrun];

@interface Animal : NSObject  
-(void)run;  
@end  
@implementation Animal  
-(void)run{      
    NSLog(@"动物会跑");      
}  
@end  
@interface Dog : Animal  
-(void)eat;  
@end  
@implementation Dog  
-(void)run{      
    NSLog(@"狗会跑");      
}  
-(void)eat{      
    NSLog(@"狗在吃");     
}  
@end  
int main(int argc, const charchar * argv[]) {  
    @autoreleasepool {         
        Animal *ani = [Animal new];  
        [ani run];  //动物在跑  
        //        [(Dog *)ani eat];(错误的)          
        Animal *an2 = [Dog new];  
        [an2 run];  //狗在跑          
        //NSObject访问了子类特有的方法 run  
        NSObject *obj = [Animal new];  
        [(Animal*)obj run]; //动物再跑          
        //访问子类特有方法  
        obj = [Dog new];  
        [(Dog *)obj run];          
        //NSObjec是所有类的基类  
        //obj可以指向任何子类          
        Dog *dog = [Dog new];          
        //注意id的使用  
        //id类型的使用  
        id obj2;  
        obj2 = ani;  
        [obj2 run];   //动物再跑          
        obj2 =dog;  
        [obj2 run];   //狗在跑        
    }  
    return 0;  

一、什么是instancetype
instancetype是clang 3.5开始,clang提供的一个关键字,表示某个方法返回的未知类型的Objective-C对象。我们都知道未知类型的的对象可以用id关键字表示,那为什么还会再有一个instancetype呢?

二、关联返回类型(related result types)
根据Cocoa的命名规则,满足下述规则的方法:

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

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

会返回一个方法所在类类型的对象,这些方法就被称为是关联返回类型的方法。换句话说,这些方法的返回结果以方法所在的类为类型,说的有点绕口,请看下面的例子:

@interface NSObject  
+ (id)alloc;  
- (id)init;  
@end  

@interface NSArray : NSObject  
@end  

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

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

按照Cocoa的命名规则,语句[NSArray alloc] 的类型就是NSArray*因为alloc的返回类型属于关联返回类型。同样,[[NSArray alloc]init] 的返回结果也是NSArray*。
三、instancetype作用
1、作用
如果一个不是关联返回类型的方法,如下:

@interface NSArray  
+ (id)constructAnArray;  
@end  

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

[NSArray constructAnArray];  

根据Cocoa的方法命名规范,得到的返回类型就和方法声明的返回类型一样,是id。
但是如果使用instancetype作为返回类型,如下:

@interface NSArray  
+ (instancetype)constructAnArray;  
@end  

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

[NSArray constructAnArray]; 

得到的返回类型和方法所在类的类型相同,是NSArray*!
总结一下,instancetype的作用,就是使那些非关联返回类型的方法返回所在类的类型!

2、好处
能够确定对象的类型,能够帮助编译器更好的为我们定位代码书写问题,比如:

[[[NSArray alloc] init] mediaPlaybackAllowsAirPlay]; //  "No visible @interface for `NSArray` declares the selector `mediaPlaybackAllowsAirPlay`"  

[[NSArray array] mediaPlaybackAllowsAirPlay]; // (No error) 

上例中第一行代码,由于[[NSArray alloc]init]的结果是NSArray*,这样编译器就能够根据返回的数据类型检测出NSArray是否实现mediaPlaybackAllowsAirPlay方法。有利于开发者在编译阶段发现错误。
第二行代码,由于array不属于关联返回类型方法,[NSArray array]返回的是id类型,编译器不知道id类型的对象是否实现了mediaPlaybackAllowsAirPlay方法,也就不能够替开发者及时发现错误。

四、instancetype和id的异同
1、相同点
都可以作为方法的返回类型

2、不同点
①instancetype可以返回和方法所在类相同类型的对象,id只能返回未知类型的对象;

②instancetype只能作为返回值,不能像id那样作为参数,比如下面的写法:

//err,expected a type  
- (void)setValue:(instancetype)value  
{  
    //do something  
}  

就是错的,应该写成:

- (void)setValue:(id)value  
{  
    //do something  
} 

猜你喜欢

转载自blog.csdn.net/qq_34476727/article/details/80463756