OC课程笔记总结8-核心语法3:构造方法

课程:核心语法


首先必须理解两个重要的内容:1,self,2,super

self,永远指向的是当前的对象,而判断当前对象到底是哪一个是通过最外层的中括号中是哪一个对象,而不是某个方法或者属性所属于的类或者对象。

这一点体现在 self = [super init] 中,super是指调用父类的方法init,这个方法返回一个self,而整个语句是 【子类对象 init】 调用了子类的init方法,而子类的init方法中有这么一句【self= 【super init】】 ,最后是return self,

因此,最外层的调用者是 子类对象,所有在这一语句中所涉及到的所有self 都是指向子类对象。不论这个self是否是在父类的方法中或是其他。super只是表示调用的是父类的方法,并不是一个指针,它只是一个

告诉编译器要调用父类方法的指示符。


构造方法是对象方法,必须由对象来调用,用于初始化对象属性。new是类方法,用于创建对象,但在实际开发中不常用new方法,因为new 方法功能太单一

要完整的创建一个可用对象实际分为两步,第一步,分配存储空间,第二部,初始化对象。在new方法中也是分别调用了两个方法完成了这两个步骤。

第一个:+alloc,用于分配空间,第二个   - init 用于初始化。在实际开发中,通常使用其他的方法来初始化,+alloc是必须要用的。开发中常常使用中括号嵌套来简化代码书写,因为可以

节省空间,不用创建那么多变量存储空间。实际 开发中创建对象的格式:

类型 *指针变量名 = 【【类型 alloc】 init】= 【类型 new】;

即 new 方法封装了alloc 和init 这两个方法。

默认的初始化值都是0,可以通过重写init方法来定义初始化的值。init 方法是属于NSObject类的对象方法。重写一个方法在implementation中重写。在编译器中,如果在implemetation中输入一个方法名

有系统提示,就说明这个方法已经声明过了。重写构造方法是有限制的,在重写的init方法中,一定要调用super init 方法,首先默认初始化一次后,才能自定义初始化。初始化很重要的一个目的是将对象中

的isa指针初始化,指向元类对象(包含类的类方法),获取方法列表。isa是指针,默认值是0,在执行了init方法之后就会初始化指针,让其指向当前类的元类(其实都是同一个类)

在重写init方法中有这么一条语句  self = [super init],这条语句可以这样理解:首先,这条代码是写在重写的init方法中的,要自定义初始化对象,必须先用默认的方法初始化一次。而子类中的重写了init方法,所以

如果用子类对象去调用init方法会变成一个死循环,但是记住,只要子类继承的了父类,一定会继承父类所有的东西,即使重写了init 方法,但是父类的init 方法还是存在于子类的存储空间中,而使用super,则表示

使用的是父类的方法,来对当前的子类对象作用,并不是对父类作用的意思。方法的作用对象还是当前的子类。即super是指调用父类的方法,并不是指向父类的对象。

重写init 方法的固定格式如下:

(id) init

{

if(self = [super init])

{

自定义初始化值;

}

return self;

}

使用 super init 会 调用父类中的init 方法,如果在父类中也重写了init 方法,即父类中也有 super init ,那么还会继续调用父类的父类的init 方法,最终知道调用了NSObject这个超类的init 方法,而NSObject

是闭源的,无法看到观法的源代码。而在使用到NSObject 中的init 方法时,会将isa指针直接指向当前的对象(而不是从父类到子类这样一层一层改变),这个指向是动态监测的。可以直接让其指向当前的子类对象。


子类从父类中所继承的所有属性在没有执行到初始化方法之前,也就是在执行了alloc之后,还没执行init,这段时间内,这个类的所有对象的属性都是默认值0,包括指针isa,此时isa并没有指向任何对象。

而在执行了init 方法后,这些属性才会有自己的值,如果没有重写init 方法,那么用到的是NSObject中的init 方法,这个方法可能只对一个属性初始化,就是isa,让其指向当前的类。


在新建子类对象时,是否分配了两块内存空间?在oc中是没有自动回收机制的。

答:不会,仍然是一块空间,super不是一个指针,并没有创建一个父类对象,参考super和self的相关知识。

如果父类和子类中有相同的方法,调用的是子类自己的方法。


self是一个指针,指向对象的isa指针,isa指针指向类对象(结构体,任何对象本质都是结构体),然后通过类对象的SEL指针(指向code区的类)去类中寻找方法,找到就实现。每个对象都有几个隐藏属性的,例如superclass,retainCount,isa等等等等。找不到方法,就通过superclass指针(指向父类的对象)找父类的,如此类推,找到NSObject都找不到,那就没有了。superclass才是指针,super 不是指针。

实际开发中创建新对象一般不写new,用 类型 ×指针变量名 = 【【类型 alloc】init】来代替new。


自定义构造方法:

重写init方法和自定义构造方法是不同的。构造方法即init 方法,自定义init方法可以加入参数,来初始化所需要的值。

init方法的命名是有规范要求的:

1.init方法一定是对象方法,用-号开头;

2.返回类型必须是id;

3.方法名用initWithxxxxx 开头,XXXX表示属性名,即表示初始化哪个属性。便于阅读。

任何代码都不重要重复做工,只要有了可以实现同一目标的代码,都可以进行复用。要降低代码之间的耦合性,而通过参数名来相联系的代码耦合性很高,所以尽量不要涉及过多的属性名,参数名。

如果只是通过方法名来耦合,如果在另一个方法中有某些属性名或者参数名修改了,并不需要修改另一个方法名中的对应属性或者参数名。

总之,降低代码耦合度,对于可以实现同一功能的代码进行多次复用,这样的代码才是优秀的代码。





















































  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值