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

原创 2004年02月26日 14:51:00

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

 

Java包设计原则

包(package)1.同一类的放到一个包中2.实体类(Bean)放到一个包中3.动作类(Action)放到一个包中4.工具类(Util)放到一个包中5.测试类单独放到一个包中,比如:com.gfte...
  • sinboy
  • sinboy
  • 2005年12月09日 12:05
  • 2455

Java设计模式六大原则或者说七大原则 整理 (其实文章里有七个。。。。)

对于Java看到过一个很有意思的说法:Java有六大心法,23种武功招式。分别就是Java设计模式六大原则和常用的23种设计模式了。本篇是对六大原则的整理。(其实文章里有七个。。。。最后一种是哈姆雷特...
  • u011288271
  • u011288271
  • 2016年09月10日 22:50
  • 4549

java类设计原则的简单运用实例一

java类设计原则的简单运用实例一
  • u013412772
  • u013412772
  • 2016年08月16日 21:52
  • 1112

类设计的5个基本原则

我们常说啥面向对象三大特性:封装,继承,多态.另一种说法是:抽象,继承,动态绑定 然后就是面向对象五大设计原则,面向对象的设计其实说到底就是类的设计嘛,没有了类就自然不能叫面向对象了.当然了像C#中还...
  • weiwenhp
  • weiwenhp
  • 2013年03月15日 03:49
  • 11204

java设计模式之三大分类,六大原则

一.设计模式定义(Design Patterns) 设计模式(Design pattern)是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。使用设计模式是为了可重用代码、让代码更...
  • xqhadoop
  • xqhadoop
  • 2017年03月11日 11:18
  • 496

数据库规范化设计的五个原则

一:表中应该避免可为空的列; 二:表不应该有重复的值或者列; 三: 表中记录应该有一个唯一的标识符    在数据库表设计的时候,数据库管理员应该养成一个好习惯,用一个ID号来 唯一的标识...
  • u013181313
  • u013181313
  • 2014年11月20日 19:43
  • 885

java SOLID原则

众所周知,Java编程最基本的原则就是要追求高内聚和低耦合的解决方案和代码模块设计,S.O.L.I.D是面向对象设计和编程(OOD&OOP)中几个重要编码原则(Programming Pricip...
  • VICHOU_FA
  • VICHOU_FA
  • 2016年09月13日 10:25
  • 1311

java设计6大设计原则

java6大设计原则: 一 : 类单一职责。         一个类只有
  • chengxu2011
  • chengxu2011
  • 2014年09月22日 10:09
  • 6865

OCP原则

 OCP原则-(Michale.Pan原创)面向对象可利用设计(OOD)的第一块基石,就是"开-闭原则(Open-Closed principle,简称OCP,大家不要误认为是ORACLE的管理员证书...
  • sophia_sy
  • sophia_sy
  • 2007年01月17日 15:11
  • 3278

Java的七大设计原则

本文转自:Java设计模式遵循的七大原则 这篇文章,作者分析有深度,带给我一些启发和思考。 最近几年来,人们踊跃的提倡和使用设计模式,其根本原因就是为了实现代码的复...
  • sugaryaruan
  • sugaryaruan
  • 2015年11月19日 12:03
  • 494
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:关于Java类设计中信息隐蔽原则的一点提示
举报原因:
原因补充:

(最多只允许输入30个字)