继承
继承的格式
格式:public class 子类名 extends 父类名 { }
范例:public class Zi extends Fu { }
继承中子类的特点:
- 子类可以有父类的内容
- 子类还可以有自己特有的内容
package com.itxuexi.it10;
public class Fu {
public void show(){
System.out.println("show类被调用");
}
}
//------------------------------------
package com.itxuexi.it10;
//继承父类
public class Zi extends Fu {
public void method(){
System.out.println("method类被调用");
}
}
//--------------------
package com.itxuexi.it10;
public class Demo {
public static void main(String[] args) {
Fu f = new Fu();
f.show();
Zi z = new Zi();
z.method();
z.show(); // 子类继承父类,可直接调用
}
}
继承好处与弊端
继承好处
提高了代码的复用性(多个类相同的成员可以放到同一个类中)
提高了代码的维护性(如果方法的代码需要修改,修改一处即可)
继承弊端
继承让类与类之间产生了关系,类的耦合性增强了,当父类发生变化时子类实现也不得不跟着变化,削 弱了子类的独立性
继承的应用场景:
使用继承,需要考虑类与类之间是否存在is…a的关系,不能盲目使用继承 is…a的关系:谁是谁的一种,例如:老师和学生是人的一种,那人就是父类,学生和老师就是子类
继承中成员访问特点
super关键字
关键字 | 访问成员变量 | 访问构造方法 | 访问成员方法 |
---|---|---|---|
this | this.成员变量 访问本类成员变量 | this(…) 访问本类构造方法 | this.成员方法(…) 访问本类成员方法 |
super | super.成员变量 访问父类成员变量 | super(…) 访问父类构造方法 | super.成员方法(…) 访问父类成员方法 |
package com.itxuexi.it10;
public class Fu {
//年龄
public int age =40;
}
//-----------------------------------------
public class Zi extends Fu {
public int age = 20;
public void show(){
int age = 30;
// 访问本方法内部的成员变量age
System.out.println(age); // 调用时,输出结果为:30
// 访问本类的成员变量age
System.out.println(this.age); // 调用时,输出结果为:20
//访问父类的成员变量age
System.out.println(super.age); // 调用时,输出结果为:40
}
}
继承中成员访问特点
在子类方法中访问一个变量
- 子类局部范围找 (子类方法内部找)
- 子类成员范围找
- 父类成员范围找
- 如果都没有就报错(不考虑父亲的父亲…)
继承中成员方法的访问特点
在子类方法中访问一个方法
- 子类成员范围找
- 父类成员范围找
- 如果都没有就报错(不考虑父亲的父亲…)
继承中构造方法的访问特点
子类中所有的构造方法默认都会访问父类中无参的构造方法
为什么呢?
- 因为子类会继承父类中的数据,可能还会使用父类的数据。所以,子类初始化之前,一定要先完成父类数据的初始化 --(父类的无参构造方法)
- 每一个子类构造方法的第一条语句默认都是:super()
如果父类中没有无参构造方法,只有带参构造方法,该怎么办呢?(否则会报错) - 通过使用super关键字去显示的调用父类的带参构造方法
- 在父类中自己提供一个无参构造方法
推荐:自己给出无参构造方法
package com.itxuexi.it10;
public class Fu {
//年龄
public Fu(){
System.out.println("Fu中无参构造方法被使用");
}
public Fu( int age){
System.out.println("Fu中带参构造方法被使用");
}
}
//------------------------------------------------------------
package com.itxuexi.it10;
public class Zi extends Fu {
public Zi(){
System.out.println("Zi中无参构造方法被使用");
}
public Zi( int age){
System.out.println("Zi中带参构造方法被使用");
}
}
//---------------------------------------------
package com.itxuexi.it10;
public class Demo {
public static void main(String[] args) {
// 子类中所有的构造方法默认都会访问父类中无参的构造方法
// 每一个子类构造方法的第一条语句默认都是:super()
Zi z = new Zi(); //Fu中无参构造方法被使用
//Zi中无参构造方法被使用
//
Zi z2 = new Zi(12); //Fu中无参构造方法被使用
//Zi中带参构造方法被使用
}
}
方法重写
方法重写概念
子类出现了和父类中一模一样的方法声明(方法名一样,参数列表也必须一样)
方法重写的应用场景:
当子类需要父类的功能,而功能主体子类有自己特有内容时,可以重写父类中的方法,这样,即沿袭了 父类的功能,又定义了子类特有的内容 。
Override注解:
用来检测当前的方法,是否是重写的方法,起到【校验】的作用
package com.itxuexi.it10.phone;
/*
手机类
*/
public class Phone {
public void call(String name){
System.out.println("给"+name+"打电话");
}
}
//---------------------------------------
package com.itxuexi.it10.phone;
/*
重写方法
*/
public class newPhone extends Phone{
//重写手机类call方法; @Override,可用来检验方法重写是否正确
@Override
public void call(String name){
System.out.println("先开启视频通讯");
//调用父类中的打电话功能;可加可不加。
super.call(name);
}
}
//----------------------------------------------------
package com.itxuexi.it10.phone;
/*
手机测试类
*/
public class PhoneDemo {
public static void main(String[] args) {
Phone p = new Phone();
p.call("赵宁");
newPhone np =new newPhone();
np.call("杨欢"); //调用父类
}
}
方法重写的注意事项
-
私有方法不能被重写(父类私有成员子类是不能继承的)
-
子类方法访问权限不能更低(public > 默认 > 私有)
package com.itxuexi.it10.jicheng01;
public class Fu {
//私有方法
private void show(){
System.out.println("Fu中show()方法被调用");
}
public void method(){
System.out.println("Fu中的method()方法被调用");
}
void cell(){
System.out.println("Fu中的cell()方法被调用");
}
}
//-------------------------------
package com.itxuexi.it10.jicheng01;
public class Zi extends Fu {
/*
父类私有方法,不能被重写 或 继承;
@Override
private void show(){
System.out.println("Zi中的show()方法被调用");
}
*/
/*
//默认 小于 父类中的 public 权限,报错
@Override
void method(){
System.out.println("Zi中的method()方法被调用");
}
*/
@Override
void cell(){
System.out.println("Zi中的cell()方法被调用");
}
}
java继承的注意事项
Java中类只支持单继承,不支持多继承
错误范例:
class A extends B, C { } //错误,不支持
Java中类支持多层继承
范例:
class A extends B { } //A继承B,同时继承C
class B extends C { }