美文网首页iOS开发技术部落iOS开发精华专题程序员
iOS大神留步,你确定你会使用id和instancetype?

iOS大神留步,你确定你会使用id和instancetype?

作者: 饭饭男 | 来源:发表于2016-12-26 14:08 被阅读664次

    写了那么多代码,懂了那么多道理,可是我就问你一句:id和instancetype的使用你懂了吗?

    1.
    - (instancetype)init{
        self = [super init ];
        if (self) {
            
        }
        return self;
    }
    
    2.
    - (instancetype)initWithFrame:(CGRect)frame{
        self = [super initWithFrame:(CGRect)frame];
        if (self) {
            
        }
        return self;
    }
    

    上述的两处代码我们经常在使用,但是有多少人知道这里的instancetype返回值表示什么意思?这里为什么用的instancetype而不用id?这里用id又会怎么样?

    要搞懂id 和instancetype的异同 首先要弄懂iOS中两个概念:关联返回类型/非关联返回类型
    根据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*。
    到这里还没看懂的[捂脸][捂脸][捂脸],没关系我也是看了两遍才看懂什么意思,多看几遍就懂了。
    

    既然关联返回类型是类方法中以alloc或new开头 实例方法中以autorelease,init,retain或self开头的,那么非关联返回类型肯定就是相反的定义了。举个例子吧

    @interface NSArray  
    + (id)constructAnArray;  
    @end  
    根据Cocoa的命名规则,这个肯定就不是关联返回类型,那一定就是非关联返回类型了
    当我们使用如下方式初始化NSArray时:[NSArray constructAnArray]; 
    根据Cocoa的方法命名规范,得到的返回类型就和方法声明的返回类型一样,是id。(重点:因为它不符合关联返回类型命名规范)
    
    对上面的返回类型我们做一点修改
    但是如果使用instancetype作为返回类型,如下:
    @interface NSArray  
    + (instancetype)constructAnArray;  
    @end  
    我们使用instancetype做返回类型
    当使用相同方式初始化NSArray时:[NSArray constructAnArray]; 得到的返回类型和方法所在类的类型相同,是NSArray*
    

    到这里就已经可以对比出来id和instancetype的异同了
    总结一下,instancetype的作用,就是使那些非关联返回类型的方法返回所在类的类型!

    为什么要使用instancetype?
    能够确定对象的类型,能够帮助编译器更好的为我们定位代码书写问题,比如:

    [[[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  
    }  
    

    相关文章

      网友评论

      • _飞翔的大虾丶:发现一篇一摸一样的,https://www.jianshu.com/p/30fd41b7ec80
        饭饭男:对,当时有参考他的文章,算不上一摸一样吧!
      • 独木舟的木:那是不是意味着作为返回类型使用的时候,优先使用instancetype,或者干脆只用instancetype呢?
        饭饭男:@独木舟的木 那倒不是,就看你需不需要返回一个当前类类型的数据了,如果不需要那还是写id吧
      • 奈焚摩尔:id作为返回未知类型的对象以及参数,深刻的记得这一点.一直知其然而不知其所以然的在用这两个,在不确定参数类型的时候就id摆上去.....

      本文标题:iOS大神留步,你确定你会使用id和instancetype?

      本文链接:https://www.haomeiwen.com/subject/fqdjvttx.html