继承性的最大特征是解决代码的重复。
下面先通过一段程序来分析一下,为什么需要继承。
范例:要求定义两个描述人与学生的类
Person.java
Student.java
class Person {
private String name;
private int 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;
}
}
class Student {
private String name;
private int age;
private String school;
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 String getSchool() {
return school;
}
public void setSchool(String school) {
this.school = school;
}
}
以上的程序里面出现了代码的重复,在自然的关系上,学生是属于人的一种类型,并且学生与人相比描述的更加细致,范围更小。
在Java之中如果要想实现继承使用extends关键字完成,定义时的语法结构如下:
class 子类 extends 父类{}
一定要记住,子类也会被称为派生类,父类也被称为基类(超类)。
范例:实现继承
class Person { private String name; private int 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; } } class Student extends Person {// 继承了Person父类 } public class TestDemo { public static void main(String args[]) { Student stu = new Student(); stu.setName("艺兴");// Person类定义 stu.setAge(27);// Person类定义 System.out.println("姓名:" + stu.getName() + ",年龄:" + stu.getAge()); } }
现在Student类里面没有编写任何方法,但是由于其继承了Person父类,所以可以直接使用Person定义的方法。如果有需要,也可以在Student类里面定义新的扩展方法。
范例:在Student类里面增加新的功能
class Person { private String name; private int 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; } } class Student extends Person {// 继承了Person父类 private String school; public String getSchool() { return school; } public void setSchool(String school) { this.school = school; } } public class TestDemo { public static void main(String args[]) { Student stu = new Student(); stu.setName("艺兴");// Person类定义 stu.setAge(27);// Person类定义 stu.setSchool("音乐学院"); System.out.println("姓名:" + stu.getName() + ",年龄:" + stu.getAge() + ",学校:" + stu.getSchool()); } }
通过以上程序的对比就可以非常清楚的发现继承的优点:
· 子类可以直接将父类的操作继续使用,属于代码重用;
· 子类可以继续扩充属于自己的标准。
利用extends关键字在大部分的情况下都可以不用去考虑给出的继承限制(前提:按照标准格式开发),但是事实上由于要限制用户的使用,所以针对于继承也有一些属于自己的要求。
在C++语言之中具备一种概念——多继承,即:一个子类可以同时继承多个父类。
范例:错误的继承
class A{} class B{} class C extends A,B{}
基本的要求就是希望C类可以同时继承A和B两个类,但是以上的操作是不正确的。那么之所以会存在多继承实际上目的也只有一个:希望一个子类可以同时继承多个父类的操作。那么可以使用多层继承。
class A {}
class B extends A {}
class C extends B {}
相当于C是B的子类,是A的孙子类。在使用多层继承的时候并没有层数的限制,不过从开发的角度认为,不要超过三层的继承关系。
范例:观察属性
class A { private String msg; public String getMsg() { return msg; } public void setMsg(String msg) { this.msg = msg; } } class B extends A { } public class Demo { public static void main(String args[]) { B b = new B(); b.setMsg("Hello"); System.out.println(b.getMsg()); } }
在现在的B类里面一定是存在有msg属性,因为如果不存在,那么setMsg()设置的内容就不可能被保存,即:getMsg()是绝对无法进行内容输出的。
但是在B类里面不能够针对于msg属性进行访问,因为它属于私有的,只能够间接的进行私有属性的访问。
class A {
public A() {
System.out.println("A类的构造方法");
}
}
class B extends A {
public B() {
System.out.println("B类的构造方法");
}
}
public class Demo {
public static void main(String args[]) {
new B();
}
}
输出结果:
A类的构造方法
B类的构造方法
此时并没有加入任何的操作代码,但是可以发现,在实例化子类对象前,先去实例化了父类对象,以及调用了父类的无参构造方法。
那么此时对于子类构造而言,就相当于隐含了一个“super()”。
class B extends A {
public B() {
super();// 父类中有无参构造时,加与不加无区别
System.out.println("B类的构造方法");
}
}
如果说此时你的父类中没有无参构造方法了,那么就必须使用super()明确调用父类的有参构造方法。
class A {
public A(String title) {
System.out.println("A类的构造方法");
}
}
class B extends A {
public B(String title) {
super(title);// 父类中有无参构造时,加与不加无区别
System.out.println("B类的构造方法");
}
}
public class Demo {
public static void main(String args[]) {
new B("Hello");
}
}
通过观察可以发现“super()”主要是由子类调用父类中的构造方法,那么这行语句一定要放在子类构造方法的首行,这一点和this()是类似的。
疑问?既然super()和this()都放在首行,如果现在子类构造没有编写super()的话,会自动使用一个super()调用父类的无参构造,那么如果写上了this(),那么是不是就表示子类无法调用无参构造呢?
通过代码的验证:super()与this()不能够同时出现,不管子类怎么折腾,它都永恒有一个存在的前提:子类对象的构造调用前一定要先执行父类构造,为父类的对象初始化后,才轮到子类对象初始化。