Java基础教程——继承

继承

一个类 可以 继承自 另一个类;

派生的类(子类)继承父类的方法和数据成员;

关键字:子类 extends 父类。

public class 继承 {
	public static void main(String[] args) {
		王思聪 a = new 王思聪();
		System.out.println(a.money);
		a.show();
	}
}
class 王健林 {
	double money = 188.88;
	void show() {
		System.out.println("不差钱");
	}
}
class 王思聪 extends 王健林 {
}

设计示例

继承主要解决的问题是:共性抽取。

继承的特点:
|--子类可以拥有父类的“内容”
|--子类可以拥有自己的“内容”

卡车和巴士都是汽车,它们共有的特征来自汽车,他们也可以有自己的特征:

public class 继承2 {
	public static void main(String[] args) {
	}
}
class 汽车 {
	private int wheels;
	public int getWheels() {
		return wheels;
	}
	public void setWheels(int wheels) {
		this.wheels = wheels;
	}
	// ------------------------
	public void 开车() {
		System.out.println("开车了");
	}
	public void 刹车() {
		System.out.println("刹车");
	}
}
class 卡车 extends 汽车 {
	private int carryingCapacity;
	public int getCarryingCapacity() {
		return carryingCapacity;
	}
	public void setCarryingCapacity(int carryingCapacity) {
		this.carryingCapacity = carryingCapacity;
	}
	// ------------------------
	public void 挂箱() {
		System.out.println("挂箱");
	}
}
class Bus extends 汽车 {
	private int seatings;
	public int getSeatings() {
		return seatings;
	}
	public void setSeatings(int seatings) {
		this.seatings = seatings;
	}
	// ------------------------
	public void 报站(String str站名) {
		System.out.println(str站名 + "已到");
	}
}

继承控制

protected:继承即可用,即使不在同一个包中

private:不给继承

this和super

this:自己的
super:父类的
|-|-super():父类构造方法
|-|-super.XXX:父类的成员(方法或变量)

public class This_and_Super {
	public static void main(String[] args) {
		S _s = new S();
		_s.m();// 直接通过子类对象,调用的是子类的方法
		_s.newNethod();
	}
}
class F {
	protected int a = 1;
	protected void m() {
		System.out.println("F");
	}
}
class S extends F {
	protected int a = 99;
	protected void m() {
		System.out.println("S");
	}
	protected void newNethod() {
		System.out.println(super.a);
		System.out.println(this.a);
		System.out.println(a);// 直接调用,是自己的值
		// ---------
		super.m();
		this.m();
		m();// 直接调用,是自己的方法
	}
}

super():再论构造方法

super必须有,不写也默认,必须第一句,只能有一个

​ (1)默认调父类的无参构造方法,直到显式调用父类的有参构造方法。

​ 详解:无论子类构造方法带不带参数,默认都是执行父类的无参构造方法。【隐藏一句代码:super()】
如果非要执行父类带参构造方法的话,使用super(参数列表)。

​ (2)super(父类构造方法)只能放在构造方法中,且必须放在第一句,因此也不能多次调用父类构造方法。

测试:

public class 构造顺序 {
	public static void main(String[] args) {
		new S();
		new S(1);
		new S(1L);
	}
}
class F {
	F() {
		System.out.println("父类构造方法");
	}
	F(int a) {
		System.out.println("父类构造方法(带参)");
	}
}
class S extends F {
	S() {
		System.out.println("子类构造方法\n");
	}
	S(int a) {
		System.out.println("子类构造方法(带参)\n");
	}
	S(long a) {
		super(1);
		System.out.println("子类构造方法2(带参),显示调用父类的带参构造方法\n");
	}
}

父类构造方法
子类构造方法

父类构造方法
子类构造方法(带参)

父类构造方法(带参)
子类构造方法2(带参),显示调用父类的带参构造方法

this()

this(...)也可以调用自己的其它构造方法,也必须放在第一句,一个构造方法中只能写一句。

class S2 extends F {
	S2() {
		this(100);
	}
	S2(int a) {
		this(100, 200);
	}
	S2(int a, int b) {
	}
}

思考题:总结super和this的用法

重写·Override

Override:方法重写、方法覆盖。
子类可以重新定义继承自父类的方法。子类方法中,可通过super来调用父类的方法。

  • 两同:方法名相同,形参列表相同
  • 两不大:返回类型不比父类大(大手大脚败家子),异常不比父类大(不得扩大范围)
  • 一不小:访问权限不比父类小(不得废公为私)

public class 重写 {
	public static void main(String[] args) {
		Ostrich os = new Ostrich();
		os.twitter();
		os.fly();
	}
}
class Bird {
	public void twitter() {
		System.out.println("wakuwaku");
	}
	public void fly() {
		System.out.println("飞");
	}
}
// 鸵鸟类
class Ostrich extends Bird {
	@Override// 不写也是重写,写了会做检测,若方法在父类不存在将报错
	public void fly() {
		super.fly();
		System.out.println("?并不会");
	}
}

wakuwaku

?并不会

返回类型不比父类大(注:Object是String的父类)

访问权限不比父类小(尧→舜→禹,启结束了禅让制度。)

class Bird {
	public Object twitter() {
		return null;
	}
	public String fly() {
		return null;
	}
}
// 鸵鸟类
class Ostrich extends Bird {
	public String twitter() {// 父类是Object,子类是String,没问题
		return null;
	}
	@Override// 父类返回类型String,改为Object
	public Object fly() {// The return type is incompatible with Bird.fly()
		return null;
	}
}
// 鸽子类
class Pigeon extends Bird {
	@Override
	private String fly() {// 父类是public,改为private
		// Cannot reduce the visibility of the inherited method from Bird
		return null;
	}
}

重写的应用场景:

对于已经存在的类,尽量不要进行代码修改,而是派生一个新的子类,对响应的方法进行重写。

final(不准修改)

Final修饰类:该类不可被继承(比如String类)
Final修饰方法:该方法不能被重写
Final修饰变量:不能被重新赋值

Java里的“常量”:

public static final int CHANG_LIANG = 10;

继承的坏处

继承是实现类复用的重要手段,但是继承会破坏封装。
每个类都应该封装其内部信息和实现细节,仅暴露必要的方法给其他类使用。但子类可以直接访问父类的内部信息,甚至可以重写父类的方法,这增加了子类和父类的耦合度,破坏了封装。

设计父类时,应遵循以下原则:

  • 尽量将父类的成员变量设置为private。
  • 尽量缩小方法的访问权限,能私有尽量不要共有。
  • 不希望被子类重写的方法设置为final。
  • 不要在父类的构造方法中调用被子类重写的方法。

​ 下例代码中,父类A的构造方法,该方法调用了被重写的method()方法,因为是先执行父类构造方法,而其调用的method()方法是被子类重写的,此时并没有变量s,因此s为null。

class A {
	public A() {
		System.out.println("A的构造方法");
		method();
		this.method();
	}
	public void method() {
		System.out.println("A method");
	}
}
class B extends A {
	public String s = "子类变量";
	public void method() {
		System.out.println("B method:" + s);
	}
}
public class TestInstanceOf {
	public static void main(String[] args) {
		B b = new B();
	}
}
posted @ 2019-07-13 01:17  虎老狮  阅读(282)  评论(0编辑  收藏  举报