Java学习笔记-对象和类
关于Java类设计中信息隐蔽原则的一点提示
作为类的属性要注意一点,不能够返回一个引用到一个一个可变对象,如果确实需要,那么必须首先克隆它。 比方说,GregorianCalendar是一个可变对象,如: class Employee { private java.util.GregorianCalendar hireDay; public java.util.GregorianCalendar getHireDay() { return hireDay; }
public void setHireDay(java.util.GregorianCalendar hireDay) { this.hireDay = hireDay; } }
class MyCalendar { Employee employee=new Employee(); GregorianCalendar gc=new GregorianCalendar(1976,5,22); public void run(){ employee.setHireDay(gc); gc.add(Calendar.YEAR,-10); System.out.println(employee.getHireDay().get(Calendar.YEAR)); } } 上面的程序断输出的是什么呢?你也许认为是1976,但是不对,结果为1966.因为修改gc的同时修改了Employee的局部变量hireDay,因为他们指向同一个地址空间。
当然上面的程序断可以改变为: class Employee { private java.util.GregorianCalendar hireDay; public java.util.GregorianCalendar getHireDay() { return (GregorianCalendar) hireDay.clone(); }
public void setHireDay(java.util.GregorianCalendar hireDay) { this.hireDay = (GregorianCalendar) hireDay.clone(); } } 这里必须说明set和get都必须使用clone()方法,否则达不到效果,只有这样才能遵循信息隐蔽的原则,不至于出现外部的引用直接修改类的属性值.
《Java 核心技术 卷 I:原理》(原书第五版)P113页上说,Date是不可变对象,但是测试是不对的,如果使用Date作为字段属性,也必须使用clone()方法否则页可以被外部引用修改,参考下面的程序段: class Employee { private java.util.GregorianCalendar hireDay; private java.util.Date birthday; public java.util.GregorianCalendar getHireDay() { return (GregorianCalendar) hireDay.clone(); }
public void setHireDay(java.util.GregorianCalendar hireDay) { this.hireDay = (GregorianCalendar) hireDay.clone(); }
public java.util.Date getBirthday() { return (Date) birthday.clone(); }
public void setBirthday(java.util.Date birthday) { this.birthday = (Date) birthday.clone(); }
}
class MyCalendar { Employee employee = new Employee(); GregorianCalendar gc = new GregorianCalendar(1976, 5, 22); Date date=new Date(1976,5,22); public void run() { employee.setHireDay(gc); gc.add(Calendar.YEAR, -10); System.out.println(employee.getHireDay().get(Calendar.YEAR)); gc=employee.getHireDay(); gc.add(Calendar.YEAR,5); System.out.println(employee.getHireDay().get(Calendar.YEAR));
employee.setBirthday(date); date.setYear(1955); System.out.println(employee.getBirthday().getYear()); date=employee.getBirthday(); date.setYear(1900); System.out.println(employee.getBirthday().getYear()); } } 以上输出全部都是1976,如果去掉任何一个clone()方法,输出都不是这样的。而我们的目的是,希望类的属性只能由类的set设置器类修改,不希望通过其它途径来修改。
以上是我阅读该书这一章的一点体会,因为发现书中由错误,所以吧体会写出来,供大家参考,这个问题我也是第一次接触到,可能我的理解也有错误,希望大家提出意见。
曾青松 [email protected] 2004-2-26

|