关于Java类设计中信息隐蔽原则的一点提示

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设置器类修改,不希望通过其它途径来修改。

以上是我阅读该书这一章的一点体会,因为发现书中由错误,所以吧体会写出来,供大家参考,这个问题我也是第一次接触到,可能我的理解也有错误,希望大家提出意见。

曾青松
zengqingsong@sohu.com
2004-2-26

 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值