Java入门系列-16-继承

这一篇文章教给新手学会使用继承,及理解继承的概念。掌握访问修饰符、掌握 final 关键字的用法。

继承

为什么要使用继承

首先我们先看一下这两个类:

public class Teacher {
	private int teachingAge;
	private String name;
	private int age;
	
	public void teach() {
		
	}
	public void seyHi() {
		System.out.println("我是:"+this.name);
	}
}
public class Student {
	private int studentNo;
	private String name;
	private int age;
	
	public void learn() {
		
	}
	public void seyHi() {
		System.out.println("我是:"+this.name);
	}
}

Student 类和 Teacher 类中有一些相同的属性和方法,这些都属于重复代码,当一个程序中有大量的类时,就会产生大量的重复代码。这些重复的代码能不能抽取出来然后供其他类使用以简化呢,那就是使用继承

使用继承优化之后:

创建 inherit 包

父类:(公共代码类)

package inherit;

public class People {
	private String name;
	private int age;
	
	public void sayHi() {
		System.out.println("我是:"+this.name);
	}

	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;
	}
	
}

子类:Student.java

package inherit;

public class Student extends People{
	private int studentNo;
	public void learn() {
		System.out.println("学习课程");
	}
	public int getStudentNo() {
		return studentNo;
	}
	public void setStudentNo(int studentNo) {
		this.studentNo = studentNo;
	}
}

子类:Teacher.java

package inherit;

public class Teacher extends People{
	private int teachingAge;
	
	public void teach() {
		System.out.println("教授课程");
	}
	
	public int getTeachingAge() {
		return teachingAge;
	}

	public void setTeachingAge(int teachingAge) {
		this.teachingAge = teachingAge;
	}
	
}

测试类:

package inherit;

public class TestInherit {
	public static void main(String[] args) {
		//创建Student对象
		Student stu=new Student();
		stu.setName("张三");//父类中继承过来的方法
		stu.learn();//子类中特有的方法
		stu.sayHi();
		//创建Teacher对象
		Teacher teacher=new Teacher();
		teacher.setName("汤尼");
		teacher.setTeachingAge(2);//子类中特有的方法
		teacher.sayHi();
	}
}

观察上面示例代码我们发现:

1.子类的公共代码都可以放在父类中

2.子类可以有自己独有的方法和属性

3.子类一旦继承父类就会拥有父类的属性和方法

4.将公共代码放入父类,更方便统一修改代码

继承的语法

关键字:extends

1.编写父类

public class 父类{
    //公共的属性和方法
}

2.编写子类,继承父类

public class 子类 extends 父类{
    //子类特有的属性和方法
}

子类只能继承一个父类

子类访问父类成员

子类要想访问父类的成员要使用 super 关键字,super 代表父类对象

访问父类构造方法:

super();//访问无参构造
super(参数);//访问有参构造

访问父类属性:

super.name;

访问父类方法:

super.print();

访问父类构造,必须在子类构造方法中调用,必须是第一句

super 只能出现在子类的方法和构造方法中

super 不能访问父类的 private 成员

敲一敲:访问父类成员

创建包 visitparent 后在报下创建如下类
父类

package visitparent;

public class Animal {
	private String name;
	private int legs;
	
	public Animal() {
		this.name="无名";
		this.legs=4;
	}
	public Animal(String name,int legs) {
		this.name=name;
		this.legs=legs;
	}
	
	public void eat(String food) {
		System.out.println(name+" 吃食物:"+food);
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public int getLegs() {
		return legs;
	}

	public void setLegs(int legs) {
		this.legs = legs;
	}
	
}

子类

package visitparent;

public class Cat extends Animal{
	private String hairColor;//毛发颜色
	private int age;
	
	public Cat () {
		super();//调用父类无参
	}
	public Cat(String name,int legs,String hairColor,int age) {
		super(name, legs);//这里调用相当于重用父类构造方法了
		this.hairColor=hairColor;
		this.age=age;
		//super(name, legs);//去掉注释试试
		//this.name="无名";//去掉注释试试
	}
	public void catchMouse() {
		System.out.println(super.getName()+":抓老鼠");
	}
	
	public void paly() {
		System.out.println(super.getName()+" 玩累了。");
		super.eat("小鱼干");
	}

	public String getHairColor() {
		return hairColor;
	}

	public void setHairColor(String hairColor) {
		this.hairColor = hairColor;
	}

	public int getAge() {
		return age;
	}

	public void setAge(int age) {
		this.age = age;
	}
	
}

子类不能继承父类的 private 成员

子类不能继承不同包使用默认访问权限的成员(默认访问权限就是不写访问修饰符)

子类不能继承父类的构造方法

多重继承的执行流程

在创建子类的时候父类在做什么?

下面创建3个类观察执行流程,C类继承B类,B类继承A类。

A.java

public class A {
	public A() {
		System.out.println("A类的无参构造函数执行");
	}
}

B.java

public class B extends A{
	public B() {
		System.out.println("B类的无参构造函数执行");
	}
}

C.java

public class C extends B{
	public C() {
		System.out.println("C类的无参构造函数执行");
	}
}

TestRunFlow.java 测试类,展示运行结果

public class TestRunFlow {
	public static void main(String[] args) {
		C c=new C();
	}
}

运行结果为:

A类的无参构造函数执行
B类的无参构造函数执行
C类的无参构造函数执行

如果子类构造方法通过 super 显式调用父类相应构造方法,则不执行父类无参构造方法

子类构造方法默认会调用父类无参构造方法

调用父类无参构造方法,一直到执行顶级父类Object类的的无参构造方法为止

根据以上规则,判断下面的代码是否能编译通过

父类

public class Pet {
	private String name;
	public Pet(String name) {
		this.name=name;
	}
}

子类

public class Dog extends Pet{
}

答案是不能,父类中只有有参构造方法没有无参构造方法,子类中没有任何代码默认有一个隐式无参构造方法,子类无参构造方法默认调用父类无参构造方法,然而父类中没有,所有在子类中报错。

解决办法:1.在父类中显式添加无参构造方法,2.在子类构造方法中显式调用父类有参构造方法。

java 中的访问修饰符

访问修饰符 protected 能修饰属性和方法,修饰后本类、子类、同包可以访问。

访问修饰符 本类 同包 子类 其他
private
默认(friendly)
protected
public

方法重写

在"继承优化后"的代码中,Teacher 继承了 People 类,(忘记代码可以翻回去再看一遍) People 类中有个一个打招呼的方法 sayHi() 用于输出人的名字,但是 Teacher 调用这个方法并不能打印出 Teacher 的属性 teachingAge 的值,但是我们还想用这个方法实现这个功能,应该怎么办呢?

我们可以使用 方法重写 解决这个问题,修改子类 Teacher 中的代码,下面看一下使用方法重写后的效果。

Teacher.java

package inherit;

public class Teacher extends People{
	//省略其他属性
	
	@Override
	public void sayHi() {
		System.out.println("我是:"+super.getName()+" ,从事教育行业 "+this.teachingAge+" 年了。");
	}

	//省略其他方法、getter、setter
}

在 Eclipse 中重写某方法的快捷键是 Alt+Shift+S+V ,按完后选择要重写的方法

在 Idea 中重写某方法的快捷键是 Ctrl+O ,按完后选择要重写的方法

@Override 注解的作用, 用来检测是否符合重写规则,不符合重写规则将报错,这个注解可以不写

构造方法不能重写,因为构造方法不能被继承

方法重写的规则:

1.方法名相同

2.参数列表相同

3.返回值类型相同或者是其子类

4.访问权限不能严于父类

final 关键字的使用

1.final 修饰变量后变为常量

private static final long serialVersionUID = -6849794470754667710L;

2.final 修饰类后,该类不能被继承

package java.lang;
public final class Math {
    //省略属性和方法……
}

3.final 修饰方法后,该方法不能被重写

public final void teach() {
		System.out.println("教授课程");
	}

4.final 修饰创建的对象后,该对像不能再次实例化(可以修改属性)

final Teacher teacher=new Teacher();
teacher.setName("汤尼");
//teacher=new Teacher();//去掉注释试试

String 类就是一个典型的被 final 修饰的类

搜索关注公众号「享智同行」,第一时间获取技术干货

posted @ 2018-10-19 21:12  享智同行  阅读(198)  评论(0编辑  收藏  举报
我要啦免费统计