对象的实例化过程:
1.检查类是否加载,如果没有加载就加载这个类。要加载所有父类(读取.class文件)
2.分配父类型空间,按照属性类型自动分配默认空间并赋给默认值
3.执行父类属性的初始化代码,如int a = 1
4.执行父类构造器代码
5.父类构造器结束,返回子类,分配子类空间,按照属性类型自动分配默认空间并赋给默认值
6.执行子类属性的初始化代码,如int b = 2
7.执行子类构造器代码
Demo:
public class PersonDemo {
/**
* @param args
*/
public static void main(String[] args) {
Student s = new Student(88);
Person p = s;
//调用方法
System.out.println(s.getName()+","+p.getName());//a student,a student
//直接调用属性
System.out.println(s.name+","+p.name);//a student,is Person
}
}
class Person{
String name = "noname";
public Person(){
name = "is Person";
}
public String getName(){
return this.name;
}
}
class Student extends Person{
String name ;
public Student(int age){
this.name = "a student";
}
public String getName(){
return this.name;
}
}
第一条输出语句输出的是:a student,a student
子类Student继承父类Person并且又覆盖了Person类的getName()方法。而Person对象p引用的是Student类的地址。所以p.getName()和s.getName()调用都是子类Student的getName()方法。而子类Student类中的方法getName()中的this.name指的也Student的属性name.而name在Student的构造函数中被赋上了值“a student”.故最后输出的都a student
第二条输出语句输出的是:a student,is Person
属性只能被继承,没有覆盖的概念。属性是绑定到类型的。当子类Student和父类Person有同名同类型的属性name时,p.name和s.name是不同的属性.它们的属性名字相同,都是name.按照就"近"原则,p到Person类中找名为name的属性.s到Student类中找名为name的属性.p子类Student对象s.name得到的是子类Student的赋给name的值“a student”.父类对象p.name得到的是父类Person赋给name的值"a person".