1.子类构造方法
先有父再有子,所以子类对象构造时,需要先调用基类构造方法,然后执行子类的构造方法。
public class Base{
public Base(){
System.out.println("Base()");
}
}
public class Derived extends Base{
public Derived(){
// super(); // 子类构造方法中默认会调用基类的无参构造方法:super();
// 用户没有写时,编译器会自动添加
System.out.println("Derived()");
}
}
执行结果如下:
从上面代码看出,在子类构造方法中,并没有写任何关于基类构造的代码,但是在构造子类对象时,先执行基类的构造方法,然后执行子类的构造方法,因为:子类对象中成员是有两部分组成的,基类继承下来的以及子类新增加的部分 。是先有父再有子,所以在构造子类对象时候 ,先要调用基类的构造方法,将从基类继承下来的成员构造完整 ,然后再调用子类自己的构造方法,将子类自己新增加的成员初始化完整 。
注意:
1. 若父类显式定义无参或者默认的构造方法,在子类构造方法第一行默认有隐含的super()调用,即调用基类构造方法。
2. 如果父类构造方法是带有参数的,此时需要用户为子类显式定义构造方法,并在子类构造方法中选择合适的父类构造方法调用,否则编译失败。
3. 在子类构造方法中,super(...)调用父类构造时,必须是子类构造函数中第一条语句。
4. super(...)只能在子类构造方法中出现一次,并且不能和this同时出现。
2.初始化
已知在没有继承时,静态代码块先执行,并且只执行一次,在类加载阶段执行 。当有对象创建时,才会执行实例代码块,实例代码块执行完成后,最后构造方法执行。
那么在继承的情况下,这些代码块的执行顺序是怎样的?
class Person {
public String name;
public int age;
public Person(String name, int age) {
this.name = name;
this.age = age;
System.out.println("Person:构造方法执行");
}
{
System.out.println("Person:实例代码块执行");
}
static {
System.out.println("Person:静态代码块执行");
}
}
class Student extends Person{
public Student(String name,int age) {
super(name,age);
System.out.println("Student:构造方法执行");
}
{
System.out.println("Student:实例代码块执行");
}
static {
System.out.println("Student:静态代码块执行");
}
}
执行结果如下:
从执行结果,我们可以得出以下结论:
1、父类静态代码块优先于子类静态代码块执行,且是最早执行。
2、父类实例代码块和父类构造方法紧接着执行。
3、子类的实例代码块和子类构造方法紧接着再执行。
4、第二次实例化子类对象时,父类和子类的静态代码块都将不会再执行。
3.protected 关键字
在类和对象的学习中,为了实现封装特性,Java中引入了访问限定符,主要限定:类或者类中成员能否在类外或者其他包中被访问。
父类private成员在相同包子类中不可见,在不同包其他类中不可见,在不同包子类中不可见。
父类中protected成员在相同包子类中可以直接访问,protected修饰的成员在不同包子类中可以直接访问,在不同包其他类中不能直接访问。
父类中public成员在相同包子类中可以直接访问,public修饰的成员在不同包子类中可以直接访问,在不同包其他类中可以直接访问。
父类中默认访问权限修饰的成员在相同包子类中可以直接访问,父类中默认访问权限修饰的成员在不同包子类中不能直接访问,父类中默认访问权限修饰的成员在不同包其他类中不能直接访问 。
4.继承方式
在Java中只支持以下几种继承方式:单继承、多层继承、不同类继承同一个类。
注意:Java中不支持多继承。
在写代码时尽量不要让类之间的继承层次太复杂,一般不要出现超过三层的继承关系,果继承层 次太多, 就需要考虑对代码进行重构了。