对象类型的转换
正像有时候需要浮点型转换为整型一样,有时候也需要将某个类的对象引用转换为另外一个类的对象引用。
在Java中我们经常看到的new操作符的表达式是:
类型 类型变量= new 类型构造函数();
其实这里我们可以分成两部分来看,
第一部分:类型 类型变量,这个跟我们声明一个数值类型没什么区别都是声明了一个变量用来存储。
第二部分:new 类型构造函数();这个是创建了一个类型的示例,
这两部分结合起来,就是我们声明了一个类型变量,说明了这个变量的类型是什么,而他又去引用了哪个类型的实例。所以变量中只是引用了对象实例。
所以我们也可以用想数值中的强制类型转换一样的方法来强制类型转换对象类型,但是这个是有先决条件的。
下面我们来做个示例说明:
首先我们两建立两个类一个父类Employee一个子类Manager
Employee.java代码如下:
public class Employee {
private String name;
private double salary;
private LocalDate hireDay;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public double getSalary() {
return salary;
}
public void setSalary(double salary) {
this.salary = salary;
}
public LocalDate getHireDay() {
return hireDay;
}
public void setHireDay(LocalDate hireDay) {
this.hireDay = hireDay;
}
public Employee(String name, double salary, int year, int month, int day) {
this.name = name;
this.salary = salary;
this.hireDay = LocalDate.of(year, month, day);
}
public Employee() {
}
public void raiseSalary(double byPercent) {
double raise = salary * byPercent / 100;
salary += raise;
}
}
Manager.java代码如下:
public class Manager extends Employee {
private double bonus;
public Manager(String name, double salary, int year, int month, int day) {
super(name, salary, year, month, day);
bonus = 0;
}
public double getSalary() {
double baseSalary = super.getSalary();
return baseSalary + bonus;
}
public void setBounds(double b) {
bonus = b;
}
}
如果我们要进行强制类型转换的时候我们只需要如果数值类型的强制类型转换一样,在要转换的类型前面加上一个小括号,在括号里写上目标类型就可以了:
Manager boss=(Manager)staff;
进行类型转换的唯一目的就是:在暂时忽视对象的实际类型之后,使用对象的全部功能(属性和方法)。
在Java中,每个对象变量都属于一个类型。类型描述了这个变量所引用的以及能够引用的对象类型。将一个值存入变量时,编译器将检查是否允许该操作的执行。如果是将一个子类赋予一个父类变量,编译器是允许的。但是将一个父类的引用赋予一个子类的变量,就必须进行类型转换,这样才能够通过运行时的检查。
而且这个转换未必是都可以了。如果运行时转换发生错误,将会产生一个ClassCastException异常。如果没有捕获这个异常,那么程序就会终止。因此,我们需要使用到一个有效的方法是用instanceof操作符:
if ( staff instanceof Manager){
boss=(Manager) staff;
}
如果instanceof的最后运算为false那么转换将不会被执行。
另外还有一个null的情况。
如果一个类型变量的引用值为null那么它跟任何对象引用进行instanceof返回结果都是false。因为他没有引用任何对象。
最后做个总结:
- 只能在继承层次内进行类型转换。
- 在将超类(父类)转换为子类之前,一定要进行instanceof检查。