在Immutable 模式中,保护类的并不是synchronized,而是immutability(不可变性)。这样一来,保护类的immutability 就是程序设计人员的工作了。
代码示例:
package immutable.sample2;
public class Main {
public static void main(String[] args) {
Person alice = new Person("Alice", "Alaska");
new PrintPersonThread(alice).start();
new PrintPersonThread(alice).start();
new PrintPersonThread(alice).start();
}
}
package immutable.sample2;
public final class Person {
private final String name;
private final String address;
public Person(String name, String address) {
this.name = name;
this.address = address;
}
public String getName() {
return name;
}
public String getAddress() {
return address;
}
public String toString() {
return "[ Person: name = " + name + ", address = " + address + " ]";
}
}
package immutable.sample2;
public class PrintPersonThread extends Thread {
private Person person;
public PrintPersonThread(Person person) {
this.person = person;
}
public void run() {
while (true) {
System.out.println(Thread.currentThread().getName() + " prints " + person);
}
}
}
Person 类的字段值仅可通过构造函数来设置。类中设有引用字段值的getName方法和getAddress 方法,但并没有修改字段值的setName 方法或setAddress方法,因此Person类的实例一旦创建,其字段值就不会发生改变,这时多线程同时访问同一个实例,Person 类也是安全的。
这里有几个特殊处理,来保证Person 类的immutable 性
Person 类声明为了final 类型 | 我们无法创建Person类的子类,但却可以防止子类修改其字段值 |
name 和 address 的可见性都是private | 这两个字段都只有从该类的内部才可以访问,这也可以防止子类修改其字段值 |
name 和 address 都声明为final | 当字段被赋值一次,就不会再被赋值。这样一来,即使不小心写了赋值代码,在编译时也会有错误提示 |