self = [super init]的思考

作者: 大爱无言 | 来源:发表于2015-12-24 00:06 被阅读2464次

    个人学知识还是想尽量学的明白些,否则觉着不顺畅,虽然很早就看苹果官方文档,https://developer.apple.com/library/ios/documentation/General/Conceptual/DevPedia-CocoaCore/Initialization.html,人家就是建议这样写

    if (self = [super init]) { // equivalent to "self does not equal nil"

    date = [[NSDate date] retain];

    }

    return self;

    }

    但是自己还是有疑惑,疑惑在于两点:

    1.这样写不是把[super init]出来的父类指针赋值给了子类self了么.这样一来,我需要的子类self不就变成了父类的self了么?可是实际使用时我们为啥没出现任何问题呢?

    2.为什么要这样写?或者说这样写的意义在于哪里?

    首先阐述我理解的第一个问题,Google了,也百度了,有人对此提出疑问,却没人针对性的给与解答,可能只有问题来自自己的思考才会做出十分针对性的回答.(请允许我说的尽量直白些,这样更有利于表达清楚).查看了说的比较好的文字有以下两个,一个是国内的,一个是国外的

    1.http://forums.macrumors.com/threads/super-init-vs-if-self-super-init.532527/

    2.http://www.cnblogs.com/tangbinblog/p/4034890.html

    当然还有签名苹果关于初始化以及alloc的描述

    各有各的理解,不过我还是觉得综合说一下,加上自己的理解有助于大家的互相学习:

    要解答第一个问题先说说这个self究竟是什么,其实刚开始接触面向对象语言的时候就见过,记得大学的时候老师反复强调的就是"指向对象自己的指针",那么粗略的说这句话没错,在学习oc后咱们几乎都看到过这样的描述"self代表着当前方法的调用者",

    这里有必要说的更直白些,我们可以想想,什么时候我们会写出这个self,只有在方法里,才会写出来,所以这么说是没有问题的,我们可以简单的结合代码来看,比如对象方法里你打印self是一个内存地址,或者说是指针,类方法里呢,打印出来的是对应的类名,也就是class类型,

    "self代表着当前方法的调用者"这句话说的对,但是似乎还有疑问,当前方法的调用者究竟该如何描述它,在层层继承关系中,这个self既然是个地址也就是指针,那么它代表的是谁的指针,runtime在消息机制里又是如何和self交互的,

    那么先看看init方法的配对使用,也就是oc的惯用方法,[UIView alloc]init,其实,在任意一个对象方法里打印self,我们都会得到一个相同的地址,也就是说alloc在分配给一个子类内存的时候,self已经出现,runtime在发送消息的时候,self是隐含参数之一,self是子类接收消息的入口,咱们在写下面这样的init方法时看似调用了[super init],也就是父类的方法,

    - (instancetype)init

    {

    self = [super init];

    if (self) {

    }

    return self;

    }

    但是我们还是在子类里调用的,目前alloc在内存里也完全是按照子类需要的内存大小分配的内存空间,既然在消息机制里,需要消息的接收主体,或者说入口,同时内存里又是为子类分配的内存,所以说在子类里调用任何方法,包括用super调用父类的方法,消息的接收实体都是当前子类,super只是告诉编译器查找方法的时候再父类方法列表里 去查找.根本不可能越过这个子类把消息直接发送到父类里去.趁热打铁看下面的代码

    #import "WYControl.h"

    @implementation WYControl

    //这是父类

    - (instancetype)init

    {

    self = [super init];

    if (self) {

    }

    return self;

    }

    #import "WYSubControl.h"

    @implementation WYSubControl

    //这是继承上一个类的子类

    - (instancetype)init

    {

    self = [super init];

    if (self) {

    }

    return self;

    }

    当你把断点打在 父类方法第一句时,你会发现此时父类的打印出来的self也是子类,也就是说明了,

    父类此时的方法调用者依然是子类self,父类俨然成为了子类的一部分,在实例化子类时父类事例self是不存在的.所以回到第一个问题,子类alloc后分配出现了self,这个self ,在[super init]链中只有一个唯一的self,就是事例子类自己的self,不存在其他self,所以self = [super init]没有出现把父类self赋值给子类self的事情.所以不会有任何问题.

    阐述第二个问题:

    这样做的意义在于,我们想要用到子类的实例,比如我们想实例一个button,那么我们必将用到button继承于uiview的某些属性,如果由于某些原因[super init]在初始化他的属性时没有成功,在父类的init方法里返回了一个nil,那么我们得到的button实例的self 也就是这个实例将会同样是nil,不会傻傻的用一个残疾的self去做事情.当然还有其他情况,比如类簇等等,苹果前面的链接里说的很清楚.当然有时候我们写成了self == [super init]也是ok的,这样写的风险就是排除不了父类返回nil的情况,因为self 与[super init]的结果本来正常情况下就应该是相等的!所以大多数情况下,这个等式都成立,所以写成

    - (instancetype)init

    {

    if (self ==[super init]) {

    }

    return self;

    }

    也一样可以,但是这样写是不对的.静态分析时就会出现警告.

    如果那里说的不对请给与指出,谢谢!如果大神们有更深入底层的理解,希望也写出来.如果谁有相应的博客地址请留个言!万分感谢!!!

    相关文章

      网友评论

      • AliceJordan:您好,谢谢您的分享 但我有一点不明白,我一般都是写成self == [super init] 但是您在最后说静态分析会出警告? 有点没明白~ 纯小白
        大爱无言:@AliceJordan 如果self是nil,[super init]也返回nil,此时恒等判断成立,显然这种情况是应该不成立的,但是=就可以判断这种情况为false
        AliceJordan:@大爱无言 恩 哪天我试了来着 但是这是为什么呢
        大爱无言:是的恒等肯定是不对的,你试试静态分析,会有警告的,一种检测代码的方法,谷歌一下
      • 不要玩霸王餐:-(void)setValue:(id)value forKey:(NSString *)key
        {
        NSArray *titles = @[@"SID",@"C_CAPTION",@"C_CONTENT",@"R_RANGE",@"S_RANGE",@"R_DEPT",@"S_DEPT",@"R_RECORDER",@"S_RECORDER",@"T_RECORD",@"N_READER",@"C_SITE",@"T_SIGN",@"C_READER",@"C_FILE",@"C_SIGN_ERR_MSG",@"users",@"users_02",@"files",@"imgs",@"B_READER"];
        if (![titles containsObject:key]) {
        return;
        }

        [super setValue:value forKey:key];
        }

        -(instancetype)initWithDictionary:(NSDictionary *)dictionary {
        if (self = [super init]) {
        [self setString];
        [self setValuesForKeysWithDictionary:dictionary];
        }
        return self;
        }
        大神 我这段代码中的[super setValue:value forKey:key];这句 为什么要用super调,求解
        大爱无言:@不要玩霸王餐 用self调用一样的
        不要玩霸王餐:@大爱无言 NSObject
        大爱无言:过誉了,我就是个程序员,你这个代码所在类的父类是什么类呢?
      • 来呀快活呀哈哈:哈哈 有点找到同道中人的赶脚,都喜欢刨根问底,恨不得挖到汇编上去,😆
        大爱无言:互相学习
      • 超_iOS:那么self.delegate =self 和_delegates=self有什么区别么?大神:smile:
        大爱无言:前一个调用的是setter方法,后面的直接给变量赋值
      • Tiffany11:所以如果这样写:
        - (id)initWithA:(NSString *)a
        {
        if(a==nil)
        return nil;
        self = [super init];
        if(self ) ......

        如果这写,就回造成内存泄漏对吗?
        必须self=nil
        大爱无言:@Tiffany11 是的
        Tiffany11:@大爱无言 对的,MRC的话,就是泄露咯?
        大爱无言:如果a == nil成立,那么返回了nil,此时外界没有任何对象持有当前alloc的对象对吧,那就相当于局部变量了,就会被自动销毁吧
      • iOS_愛OS:还是喜欢这样的文章,谢谢博主
        大爱无言:共同学习
      • 散客老金:终于理解了,thx
        大爱无言:@WzDTj 互相学习!
      • 流浪_先生:很同意 这句话 : 在[super init]链中只有一个唯一的self,就是事例子类自己的self,不存在其他self,所以self = [super init]没有出现把父类self赋值给子类self的事情。说下个人理解,首先这里只有一个self对象, self父类属性的初始化也在self对象内部 ,什么super的对象指针(不存在),self的指针全部指的是该对象的初始位置,从计算机寻址角度来看,有了对象初始位置,以及对象大小,可以获取对象的所有内容,下面是我作的一个图,能够清晰的说明我的意思:http://images.cnblogs.com/cnblogs_com/markstray/830329/o_%E7%B1%BB%E5%88%9D%E5%A7%8B%E5%8C%96.png
        (突然发现 回复不能直接贴图 上面是图片的链接 复制到浏览器地址栏打开)
        大爱无言:@mark流浪 okok!互相学习!
      • 撸猫摸鱼侠:写得好!受教了
      • dose_爽:mark!!!!!
      • 51bitquant:文章不错!
        大爱无言:@StephenMark 谢谢!
      • af262d08760d:宇哥,我是你的脑残粉
        大爱无言:@饿雨 哈哈!互相学习

      本文标题:self = [super init]的思考

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