之前说了面向对象三大特性,讲过了封装,这次我们讲继承,后面讲多态
父类和子类,相信学过python也知道,父类是高级抽象共性的模型,子类继承父类的部分或全部属性方法,可以拥有其自己的属性方法
我们定义父类,就和之前的定义类一样,定义子类则需要如下格式
修饰符 class 子类名 extends 父类名{}
如上,我们定义了一个Employee类,然后Teacher extends Employee,我们在子类Teacher里虽然没有定义任何东西,但是父类有的属性方法都可以使用
我们知道子类访问实例变量使用this.,如果想在子类访问父类的成员变量,使用super.
对于子类的实例调用属性方法,如果属性是子类未定义则去父类属性去找,如果重名则使用子类的,对于方法调用,如果子类未定义则去父类方法找,如果重名且参数列表相同,使用子类的方法,方法和属性重复的叫做覆写Override
我们可以在代码里使用@Override检查是否为有效重写
如上,我们父类Employee里定义了方法method,我们在子类Teacher里定义method,同时打上标识符@Override并没任何问题,但是我们子类改名method1,就会标红,提示不是有效重写,这里可以判断是否真实的覆盖了父类防止出错
上图需要准确理解,子类复写方法必须函数名类型和参数列表一致,子类方法返回类型继承必须小于等于父类的范围,子类方法权限必须大于等于父类权限
如上3张图,我们在父类和子类的构造方法都打印,判断到底先执行哪个,实际结果是先执行父类构造方法后执行子类构造方法
其实原因就如上,子类构造方法执行时,如果没有专门写父类的构造方法,会默认给你使用super()来执行父类构造方法,而且如果你手动指定,super()必须是子类构造方法的第一个语句(而且只能出现在构造方法中,而且子类的一个构造方法只能使用一次父类构造方法),如果是有参数的子类构造方法,那需要使用super(参数)的构造方法
这里特别说下this的用法
this关键字使用场景:
1在成员方法里访问成员变量
2 在成员方法里调用成员方法,this.可省略
3 在构造方法里调用其他构造方法,本质同上,这种情况下this调用的构造方法付也必须是第一句
如上图,我们可以看见其描述的2,3使用
还需要注意的是,我们的构造方法不能互相this引用,会将造成无限递归构造
继承内存图分析
如上图,我们定义父类,里面有成员变量,方法method,子类定义成员变量,和方法,然后使用Demo类的main作为入口函数,3个类都被放在了类和方法区(这里有继承关系的隐含了子类super指向父类),main方法里,实例化子类,就会在堆中开辟空间,空间里有子类内容,同时也有父类的内容,
调用show方法,子类有,子类的show方法近战,讲究分别打印局部变量,子类成员变量,父类变量,根据类指向super区域,运行完出栈,局部变量清空。然后实例调用method方法,子类method进栈,其内部又调用父类super的method,根据指向,父类的method进栈,运行,运行完出栈,然后子类method出栈,全部运行完main方法出栈,堆空间待垃圾清理回收
java继承的3大特征
1 是单继承的,只能有一个直接父类
解释如上图,如我们写D继承A,C是错误的,如A,C都有method,我们在子类调用method那到底执行哪个父类的方法呢,java看来是种矛盾
2 java是可以多级继承的(其实是针对上面的)
这里即是爷爹孙的这种关系,如哺乳动物->灵长类->人,java里默认所有类的父类或者叫基类为java.lang.Object类,和python一样,万物皆对象,一般我们使用继承就已经是object的子类和孙子类
3 一个父类可以有很多子类,这个没什么说的