JavaSE_65_对象的构造方法以及初始化

有关类的继承的内容大概其就这么多了。之前提过了对象的构造以及初始化。现在在类的继承的基础上,继续来说对象的构造以及初始化,主要讨论子类如何完成继承父类的初始化。

 

当调用类的构造器来创建对象时,它将给新建的对象分配内存,并对对象进行初始化操作。

 

现在我们来探讨对对象进行初始化操作时候的细节。

 

 

 

对象的初始化操作将递归如下的步骤来进行:

 

 


1.设置实例变量的值为缺省的初始值 (0, false, null),不同的数据类型有不同的初始值。


2.调用类的构造器 (但是还没有执行构造方法体),绑定构造器参数。


3.如果构造器中有this()调用,则根据this()调用的参数调用相应的重载构造器,然后,转到步骤5;否则转到步骤4。


4.除java.lang.Object类外,调用父类的中的初始化块初始化父类的属性,然后调用父类构造器,如果在构造器中有super()调用,则根据super()中的参数调用父类中相应的构造器。

5.使用初始化程序和初始化块初始化成员。

6.执行构造器方法体中其他语句。

所谓的初始化块,就是我们前面提到的所谓“游离块”。不管使用哪个构造器创建对象,它都会被首先运行,然后才是构造器的主体部分被执行。


我们来看一个例子:

 

class Person {
 private String name;

 private int age;

 private String sex;

 public Person() {
  System.out.println("构造器Person()被调用");
  sex = "Male";
  System.out.println("name=" + name + " ,age=" + age + " ,sex=" + sex);
 }

 public Person(String theName) {
  // 调用构造器Person()
  this();
  System.out.println("构造器Person(String theName)被调用");
  name = theName;
  System.out.println("name=" + name + " ,age=" + age + " ,sex=" + sex);
 }

 public Person(String theName, int theAge) {
  // 调用构造器Person(String theName)
  this(theName);
  System.out.println("构造器Person(String theName,int theAge)被调用");
  age = theAge;
  System.out.println("name=" + name + " ,age=" + age + " ,sex=" + sex);
 }

 // 初始化块
 {
  name = "Tony Blair";
  age = 50;
  sex = "Female";
  System.out.println("初始化块执行后:name=" + name + " ,age="
+ age + " ,sex=" + sex);
 }
}

public class TestPerson {
 public static void main(String[] args) {
  Person person = new Person();
 }
}


编译执行上面的程序,将会得到如下的输出:
初始化块执行后:name=Tony Blair ,age=50 ,sex=Female
构造器Person()被调用
name=Tony Blair ,age=50 ,sex=Male


可以看到,初始化块会先于构造器调用执行。读者可以将main()方法中调用的创建Person对象的构造器换成其他两个,在观察它的结果。同样可以得出上面的结论。

提示:初始化块的机制并不是必须的,你完全可以将属性的初始化和属性的声明结合在一起,如:    String name =  "Tony Blair";
下面我们看一个对象初始化的例子,以加深对对象初始化的理解。

class Person {
 private String name;

 private int age;

 private String sex;

 public Person() {
  System.out.println("构造器Person()被调用");
  sex = "Male";
  System.out.println("name=" + name + " ,age=" + age + " ,sex=" + sex);
 }

 public Person(String theName) {
  System.out.println("构造器Person(String theName)被调用");
  name = theName;
  System.out.println("name=" + name + " ,age=" + age + " ,sex=" + sex);
 }

 public Person(String theName, int theAge) {
  System.out.println("构造器Person(String theName,int theAge)被调用");
  name = theName;
  age = theAge;
  System.out.println("name=" + name + " ,age=" + age + " ,sex=" + sex);
 }

 // 初始化块
 {
  name = "Tony Blair";
  age = 50;
  sex = "Female";
  System.out.println("Person初始化块执行后:name=" + name + " ,age="
+ age + " ,sex=" + sex);
 }
}


这里定义了一个父类Person,它里面定义了三个构造器以及一个初始化块。


我们再来定义一个Person类的子类Teacher,如下:

class Teacher extends Person {
 // 部门
 String department;

 // 教龄
 int schoolAge;

 public Teacher() {
  System.out.println("构造器Teacher()被调用");
 }

 public Teacher(String name) {
  // 调用父类中的构造器Person(String theName)
  super(name);
  System.out.println("构造器Teacher(String name)被调用");
 }

 public Teacher(int theSchoolAge) {
  schoolAge = theSchoolAge;
 }

 public Teacher(String dept, int theSchoolAge) {
  // 调用本类中重载的构造器Teacher(int theSchoolAge)
  this(theSchoolAge);
  department = dept;
 }

 // 初始化块
 {
  department = "教务部";
  System.out.println("Teacher初始化块执行后:name=" + name + " ,age=" + age
    + " ,sex=" + sex);
 }
}


这个类中定义了四个构造器:一个不带参数的构造器;一个带一个String数据类型参数的构造器,它通过super()显式调用父类的构造器;一个带一个int数据类型参数的构造器;一个带两个参数的构造器,通过this()来调用类中带int类型参数的构造器。


public class TestInit {
 public static void main(String[] args) {
  System.out.println("------------------------------------");
  Teacher t1 = new Teacher();
  System.out.println("");

  System.out.println("------------------------------------");
  Teacher t2 = new Teacher("Tom");
  System.out.println("");

  System.out.println("------------------------------------");
  Teacher t3 = new Teacher("财务部", 20);
 }
}


这个程序通过三种构造器来创建三个Teacher对象,因为调用的构造器不同,所以对象初始化的步骤也有所不同。因为在这几个程序中,在几个关键部分都已经有信息打印到控制台,所以,只要执行这个程序,就可以看出各个调用构造器创建对象的运行细节。


编译并运行TestInit,可以在控制台上得到如下的信息:
------------------------------------
Person初始化块执行后:name=Tony Blair ,age=50 ,sex=Female
构造器Person()被调用
name=Tony Blair ,age=50 ,sex=Male
Teacher初始化块执行后:name=Tony Blair ,age=50 ,sex=Male
构造器Teacher()被调用

------------------------------------
Person初始化块执行后:name=Tony Blair ,age=50 ,sex=Female
构造器Person(String theName)被调用
name=Tom ,age=50 ,sex=Female
Teacher初始化块执行后:name=Tom ,age=50 ,sex=Female
构造器Teacher(String name)被调用

------------------------------------
Person初始化块执行后:name=Tony Blair ,age=50 ,sex=Female
构造器Person()被调用
name=Tony Blair ,age=50 ,sex=Male
Teacher初始化块执行后:name=Tony Blair ,age=50 ,sex=Male


在讲述数据类型的时候,我们已经知道,各种简单数据类型之间是可以进行相互转换的,有些转换可以通过系统自动完成,而有些转换必须在程序中通过强制转换来完成。

 

而对于引用类型,也有一个相互转换的机制。同样的,在引用类型数据进行转换的时候,分为自动造型强制造型两种情况。


当从子类转换成父类的时候(或者实现类转换成接口,下同),造型可以自动完成,比如,Teacher是Person的子类,则如果将一个Teacher对象赋给一个Person类型的变量的时候,造型自动完成。


当从父类转换成子类的时候(或者接口转换成实现类),必须使用强制造型,比如,Teacher是Person的子类,如果需要将一个Person对象赋给一个Teacher类型变量的时候,必须使用强制造型。


对象的强制造型可以使用运算符“()”来完成,格式如下:
 SupClass sup = new SubClass();
 SubClass sub = (SubClass)sup;


需要注意的是,无论是自动造型还是强制造型,都只能用在有继承关系的对象之间

并不是任意的父类类型数据都可以被造型为子类类型,只有多态情况下,原本就是子类类型的对象被声明为父类的类型,才可以通过造型恢复其“真实面目”,否则会在程序运行时出错。


在造型的时候,往往需要使用instanceof来判断一个对象是否可以进行造型,以避免运行时的错误(Runtime error)。


比如,Person类有两个子类:Teacher 和Student,还有一个单独的类Test,我们来看下面的代码片断:
Test t = new Test();
Person p = new Student();

Teacher t = (Teacher)p;//将会出现运行时错误
Student s = (Student)t;//因为Test类和Student之间没有任何的继承关系
//所以在编译的时候就会出错(编译错误,compile error)


此时,可以先使用instanceof来判断要造型的对象是否为可以造型的类型,如:


Person p = new Student();
if(p instanceof Teacher){
 Teacher t = (Teacher)p;
}


这样就可以避免出现运行时错误。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值