继承
一.继承概述、使用继承的好处
1.什么是继承?
-
Java中提供一个关键字extends,用这个关键字,我们可以让一个类和另一个类建立起父子关系。
-
public class Student extends People {}
-
Student称为子类(派生类),People称为父类(基类 或超类)。
-
作用:当子类继承父类后,就可以直接使用父类公共的属性和方法了
-
好处:可以提高代码的复用性,减少代码冗余,增强类的功能扩展性。
2.继承的设计规范
-
子类们相同特征(共性属性,共性方法)放在父类中定义,子类独有的的属性和行为应该定义在子类自己里面。
-
如果子类的独有属性、行为定义在父类中,会导致其它子类也会得到这些属性和行为,这不符合面向对象逻辑。
二.继承的特点
①子类可以继承父类的属性和行为,但是子类不能继承父类的构造器。
②Java是单继承模式:一个类只能继承一个直接父类。
③Java不支持多继承、但是支持多层继承。
④Java中所有的类都是Object类的子类。
1.子类是否可以继承父类的构造器?
- 不可以的,子类有自己的构造器,父类构造器用于初始化父类对象。
2.子类是否可以继承父类的私有成员?
- 可以的,只是不能直接访问。
public class Test {
public static void main(String[] args) {
//理解继承的特点
//1.子类不能继承父类的构造器
//2.子类是否可以继承父类的私有?可以继承父类的私有成员,但是不能直接访问
Tiger t = new Tiger();
//t.eat;//报错
}
}
class Animals{
private void eat(){
System.out.println("动物都要吃~~~");
}
}
class Tiger extends Animals{
}
3.子类是否可以继承父类的静态成员?
-
子类可以直接使用父类的静态成员(共享)
-
但个人认为:子类不能继承父类的静态成员。(共享并非继承)
Java只支持单继承,不支持多继承。
Java支持多层继承,子类 A 继承父类 B ,父类B 可以 继承父类 C
Object特点:
- Java中所有类,要么直接继承了Object , 要么默认继承了Object , 要么间接继承了Object, Object是祖宗类。
三.继承后:成员变量、成员方法的访问特点
1.在子类方法中访问成员(成员变量、成员方法)满足:就近原则
-
先子类局部范围找
-
然后子类成员范围找
-
然后父类成员范围找,如果父类范围还没有找到则报错。
**2.如果子父类中,出现了重名的成员,会优先使用子类的,**此时如果一定要在子类中使用父类的怎么办?
可以通过super关键字,指定访问父类的成员。
格式:super.父类成员变量/父类成员方法
四.继承后:方法重写
1.什么是方法重写?
在继承体系中,子类出现了和父类中一模一样的方法声明,我们就称子类这个方法是重写的方法。
2.方法重写的应用场景
当子类需要父类的功能,但父类的该功能不完全满足自己的需求时。
子类可以重写父类中的方法。
3.案例演示:
旧手机的功能只能是基本的打电话,发信息
新手机的功能需要能够:基本的打电话下支持视频通话。基本的发信息下支持发送语音和图片。
public class Test2 {
public static void main(String[] args) {
//方法的重写
NewPhone apple = new NewPhone();
apple.call();
apple.sendMsg();
System.out.println("============");
Phone hawei = new Phone();
hawei.call();
hawei.sendMsg();
}
}
/**
新手机,子类
*/
class NewPhone extends Phone{
//重写方法
@Override// 重写校验注解,这个方法必须是正确的重写。
public void call(){
super.call();
System.out.println("开始视频通话~~~");
}
//重写方法
@Override
public void sendMsg(){
super.sendMsg();
System.out.println("发送好看的图片~~~");
}
}
/**
旧手机,父类
*/
class Phone{
public void call(){
System.out.println("打电话~~~");
}
public void sendMsg(){
System.out.println("发短信~~~");
}
}
4.@Override****重写注解
@Override是放在重写后的方法上,作为重写是否正确的校验注解。
加上该注解后如果重写错误,编译阶段会出现错误提示。
- 重写方法都加@Override注解
5.方法重写注意事项和要求:
- 重写方法的名称、形参列表必须与被重写方法的名称和参数列表一致。
私有方法不能被重写。
子类重写父类方法时,访问权限必须大于或者等于父类 (暂时了解 :缺省 < protected < public)
子类不能重写父类的静态方法,如果重写会报错的。
五.继承后:子类构造器的特点
1.子类继承父类后构造器的特点:
子类中所有的构造器默认都会先访问父类中无参的构造器,再执行自己。
2.为什么?
子类在初始化的时候,有可能会使用到父类中的数据,如果父类没有完成初始化,子类将无法使用父类的数据。
子类初始化之前,一定要调用父类构造器先完成父类数据空间的初始化。
3.怎么调用父类构造器的?
子类构造器的第一行语句默认都是:super(),不写也存在。
public class People {
private String name;
private int age;
public People(){
}
public People(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
public class Student extends People{
public Student(){
}
public Student(String name,int age){
//调用父类的有参数构造器,初始化继承父类的数据
super(name,age);
}
}
public class Test {
//子类构造器如何去访问父类有参数构造器
public static void main(String[] args) {
Student s = new Student("张飞",28);
System.out.println(s.getName());
System.out.println(s.getAge());
}
}
super调用父类有参数构造器的作用:
- 初始化继承自父类的数据。
如果父类中没有无参数构造器,只有有参构造器,会出现什么现象呢?
- 会报错。因为子类默认是调用父类无参构造器的。
如何解决?
- 子类构造器中可以通过书写 super(…),手动调用父类的有参数构造器
六.this、super使用总结
this和super详情
this:代表本类对象的引用;super:代表父类存储空间的标识。
关键字 | 访问成员变量 | 访问成员方法 | 访问构造方法 |
---|---|---|---|
this | this.成员变量 访问本类成员变量 | this.成员方法(…) 访问本类成员方法 | this(…) 访问本类构器 |
super | super.成员变量 访问父类成员变量 | super.成员方法(…) 访问父类成员方法 | super(…) 访问父类构造器 |
public class Student {
private String name;
private String schoolName;
public Student() {
}
//如果学生不填写学校,默认这个对象的学校是三国
public Student(String name){
//借用本类兄弟构造器
this(name,"三国程序员");
}
public Student(String name, String schoolName) {
this.name = name;
this.schoolName = schoolName;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getSchoolName() {
return schoolName;
}
public void setSchoolName(String schoolName) {
this.schoolName = schoolName;
}
}
public class Test {
public static void main(String[] args) {
Student s1 = new Student("张飞","蜀国");
System.out.println(s1.getName());
System.out.println(s1.getSchoolName());
Student s2 = new Student("关羽");
System.out.println(s2.getName());
System.out.println(s2.getSchoolName());
}
}
**this(…)和super(…)**使用注意点:
子类通过 this (…)去调用本类的其他构造器,本类其他构造器会通过 super 去手动调用父类的构造器,最终还是会调用父类构造器的。
注意:this(…) super(…) 都只能放在构造器的第一行,所以二者不能共存在同一个构造器中。