1.问题

/**
* 输出: Mon Apr 26 10:54:10 CST 2010
* Mon Apr 26 10:54:10 CST 2010
*/
public static void main(String[] args){
Example test = new Example(new Date());
Date d = test.getDate();
double tenYearsInMillisSeconds = 10 * 365.25 * 24 * 3600 * 1000;
d.setTime((long) (d.getTime() - tenYearsInMillisSeconds));
System.out.println(d);
System.out.println(test.getDate());
}
public class Example {
private Date date;
public Example(Date date){
this.date = date;
}
public Date getDate() {
return date;
}
}

Date类破坏了Example的封装,导致修改实例 d时影响了 test 的值,原因是Date类生成的对象是可变的。

2.对象与对象变量

Date birthday = new Date();
Date deadline = birthday;

这两个变量引用同一个对象(请参见图 4-4 )。

java 返回对象包名规范 java如何返回对象_成员变量

但一个对象变量并没有实际包含一个对象,而仅仅引用一个对象。

在 Java 中,任何对象变量的值都是对存储在另外一个地方的一个对象的引用。new 操作符的返回值也是一个引用。

Date birthday = new Date();可以理解为new Date() 构造了一个 Date 类型的对象, 并且它的值是对新创建对象的引用。这个引用存储在变量 birthday中。

Java 对象变量与 C++ 的引用并不同

可以将 Java 的对象变量看作 C++ 的对象指针。例如,

Date birthday; // Java

实际上,等同于

Date* birthday; // C++

所有的 Java 对象都存储在堆中。 当一个对象包含另一个对象变量时, 这个变量依然

包含着指向另一个堆对象的指针。

3.更改器方法与访问器方法

上文还是没有解释清楚为什么Date类的对象是可变对象,原因在这。

假设在上文中Example类中使用Java中与Date类相近的LocalDate类便不会出现上述情况,测试可以自己去尝试。

原因在于假设使用LocalDate类中的plusDays 方法来修改对象变量,它会生成一个新的LocalDate对象,然后把这个新对象赋值给调用者,原来的对象不做任何改动。

此类只访问对象而不修改对象的方法有时称为访问器方法(accessor method)

而像Date类中的setTime方法会使得原对象的状态发生改变,此类称为更改器方法(mutator method)

4.解决方法

如果需要返回一个可变数据域的拷贝,就应该使用 clone。这样会创建一个当前对象的副本,而不会对当前对象造成影响。

public class Example {
private Date date;
public Example(Date date){
this.date = date;
}
public Date getDate() {
return (Date) date.clone();
}
}

5.不可变类

5.1什么是不可变类

不可变类指当类被实例化后,该类的成员变量均不可被改变。

如JDK内部自带的很多不可变类Interger、Long、 Boolean和String等。

5.2优缺点

优点:1.线程安全 2.易于构造、使用和测试 3.可以被自由地共享

缺点:对于每一个不同的值都需要对应一个单独的对象

5.3如何实现不可变类

Class需要用final修饰,保证类不能被继承

所有成员变量需要private修饰,保证成员变量不能直接被访问

类中不允许提供setter方法,保证成员变量不会被改变

在getter方法中不能返回对象本身,返回对象的拷贝