java零基础入门-面向对象篇(九) 继承的规则
关于继承的规矩很多,再加上初学者一般不会有太深的理解,所以学起来磕磕绊绊,绕来绕去,就差死记硬背了,我一直认为死记硬背的学习方法简直就是个灾难。
学习知识应该从原理开始理解,这样就可以避免死记硬背的学习方式,我们这一章就当做一个学习方法的例子来讲解,看看我们在学习过程中如何通过理解代替死记硬背。
首先来看一个莫名其妙出场率很高的题目,重载和重写。
重载和重写
方法的重载,是在同一个类中,名称相同,参数不同的方法。而重写则是子类继承父类以后,在子类中写了一个和父类方法同名的方法。
我们继续用上一章中的vip继承来解释这个概念。
首先我们的普通玩家,可以有几种购物的方法,可以单独买一把大宝剑,也可以大宝剑和宠物一起买。而子类VIP1玩家继承了父类普通玩家,但是VIP购物是可以打折的,所以VIP重写了一个购物方法,因为他可以打折。只要我们稍微理解一下概念,就不会将重载和重写的概念混淆,所以这个问题的出场率确实很莫名其妙。
super 关键字
当子类对父类的方法进行重写以后,父类的方法就不可见了,也可以说是被覆盖了。比如上面的例子,一旦你成为了VIP,那么你买买买的时候,肯定是打折的,因为你不是普通玩家了,你无法调用普通玩家的方法。
但是,天下没有免费的午餐,也没有免费的永久VIP,VIP可是有时间限制的,当你的VIP到期了,不好意思,要么请继续充钱当VIP,要么回到普通玩家。现在我们在VIP的类中加一个到期时间,如果VIP到期,当他买买买的时候,调用的是普通玩家未打折的方法。
我们要在VIP中调用普通玩家的方法,上面不是说被覆盖了不可见么?其实我们是可以在子类中通过super关键字来调用父类被覆盖的方法的。
super和this
看了上面的例子,是不是联想到我们曾经学过的this,他们确实有点像,但是也有区别。先来看看相似之处。
相似
1.没有重名的时候,都会隐身,当有重名的时候必须现身。参数与成员变量重名,使用this,父类成员变量与子类成员变量重名,使用super。
2.他们都可以调用构造器。this调用自己类中的构造器,而super可以调用父类中的构造器。
super除了调用被子类重写后被隐藏的成员变量和方法,还有一个重要的作用就是调用父类构造器。
子类构造时,会调用父类构造器,使用super则指定调用哪一个父类的构造器,如果不使用则默认调用父类的无参构造器。在时间顺序上,先调用父类构造器,再调用子类构造器,如果直接父类还有父类,那么再优先调用其父类的构造器,依次往上。
和this一样,第一行只能出现this或者super之一,要么调用自己的构造器,要么调用父类构造器。来看看VIP对构造器的解释
这里有个概念要确定一下,构造器用来创造对象,但是这里调用父类构造器并不是创建父类的对象,而是子类在创建对象的过程中,“借用” 父类构造器创建子类的对象。这个概念请不要混淆。
区别
this指代一个对象,哪个对象在调用方法,this就是哪个对象。而super不能指代一个对象,因为当子类被实例化的时候,并没有创建父类的对象,所以super不能理解为父类的对象。
在创建子类对象的时候,并没有创建父类对象,但是由于子类继承了父类,内存中不但会划分子类的变量空间,也会将子类的所有父类的成员变量也划分内存空间,所以可以使用super去访问他们。
允许重写和阻止重写
前面说封装的时候,有一个修饰符叫protected,他是专门给子类用来重写的时候定义的。因为父类定义private的时候,无论是变量还是方法,子类都是不可见的。所以当我们有个方法是专门给子类去重写而又不希望被其他类发现的时候,我们可以定义为protected方法。
还有种情况是我们不希望子类重写父类。比如我们普通玩家打怪的时候,一不小心怪没打死,自己挂了。这时候会有系统惩罚的,比如掉经验掉金币,就算用户是VIP,我也不希望他们改变这个机制,为什么?因为你是VIP你打怪还能死,说明啥?要么太傻,要么是冲的钱还不够!所以这个死亡掉经验掉金币的方法,不论是普通玩家不论是什么等级的VIP,都不能改变。如何实现?
final 关键字,有一个用法就是阻止重写。final这个关键字就好像一把锁,加在谁面前,谁就不能再改了。加在变量前面,这个变量就变成常量了,不能再赋值了,加在方法前面,子类统统不能改。
我们在开发软件的过程中,会是一个多人配合团体合作的过程。所以当我们设计类,方法的时候,一定要考虑清楚,你可以很直观的写一个protected修饰符,告诉别的组员,我这个方法你可以写个子类随便改,也可以写一个final告诉其他人我这个方法是不能改的。这样不但可以节省沟通成本,还可以降低方法滥用造成的错误。
为什么只能有一个直接父类?
我们可以回头看看前面的定义,为什么java中只允许有一个直接的父类?
学了super,我们知道super可以调用父类的属性和方法,假如我们可以定义多个直接父类,那么我们的super在调用的时候就不知道到底调用哪个直接父类了。
有没有理解的更深点?
最后,来简单聊一下所有对象的祖宗,java.lang.Object。这个Object 是所有类的父类,不管你的类结构是什么样,他永远站在最顶端。有了它,你的类一生下来,就具有了一部分的方法,可以直接使用,具体的我会在讲解java的包结构的时候为大家具体介绍,这里先混个眼熟。