Java中的不可变类

概念:不可变类的意思是创建该类的实例后,该实例的属性是不可改变的。java中的8个包装类和String类都是不可变类。所以不可变类并不是指该类是被final修饰的,而是指该类的属性是被final修饰的。

自定义不可变类遵守如下原则:

1、使用private和final修饰符来修饰该类的属性。

2、提供带参数的构造器,用于根据传入的参数来初始化属性。

3、仅为该类属性提供getter方法,不要提供setter方法。

4、如果有必要,重写hashCode和equals方法,同时应保证两个用equals方法判断为相等的对象,其hashCode也应相等。

构造一个不可变类非常容易,下面举一个简单例子:

  1. package com.home;  
  2.   
  3. public class Address {  
  4.     private final String detail;  
  5.   
  6.     public Address() {  
  7.         this.detail = "";  
  8.     }  
  9.   
  10.     public Address(String detail) {  
  11.         this.detail = detail;  
  12.     }  
  13.   
  14.     public String getDetail() {  
  15.         return detail;  
  16.     }  
  17.   
  18.     @Override  
  19.     public int hashCode() {  
  20.         return detail.hashCode();  
  21.     }  
  22.   
  23.     @Override  
  24.     public boolean equals(Object obj) {  
  25.         if (obj instanceof Address) {  
  26.             Address address = (Address) obj;  
  27.             if (this.getDetail().equals(address.getDetail())) {  
  28.                 return true;  
  29.             }  
  30.         }  
  31.         return false;  
  32.     }  
  33.   
  34. }  


但是值得注意的是,该类的属性虽然是被final修饰的,但若属性是非String的其他引用类型的话,那么虽然该属性的内容(所指对象的地址)不会改变,但其指向的对象却有可能会改变,这样的类当然并不能成为不可变类。比如下面的Person类中有一个Name类型的属性:

  1. package com.home;  
  2.   
  3. public class Person {  
  4.     private final Name name;  
  5.   
  6.     public Person(Name name) {  
  7.         super();  
  8.         this.name = name;  
  9.     }  
  10.   
  11.     public Name getName() {  
  12.         return name;  
  13.     }  
  14.   
  15.     public static void main(String[] args) {  
  16.         Name n = new Name("三""张");  
  17.         Person p = new Person(n);  
  18.         System.out.println(p.getName().getFirstName());  
  19.         // 改变Person对象Name属性的firstName属性值  
  20.         n.setFirstName("无忌");  
  21.         System.out.println(p.getName().getFirstName());  
  22.     }  
  23. }  


Name:

  1. package com.home;  
  2.   
  3. public class Name {  
  4.     private String firstName;  
  5.     private String lastName;  
  6.   
  7.     public Name() {  
  8.         super();  
  9.     }  
  10.   
  11.     public Name(String firstName, String lastName) {  
  12.         super();  
  13.         this.firstName = firstName;  
  14.         this.lastName = lastName;  
  15.     }  
  16.   
  17.     public String getFirstName() {  
  18.         return firstName;  
  19.     }  
  20.   
  21.     public void setFirstName(String firstName) {  
  22.         this.firstName = firstName;  
  23.     }  
  24.   
  25.     public String getLastName() {  
  26.         return lastName;  
  27.     }  
  28.   
  29.     public void setLastName(String lastName) {  
  30.         this.lastName = lastName;  
  31.     }  
  32. }  


运行上面程序可以看到,Person对象的Name属性的firstName属性已经被改变,这就违背了不可变类设计的初衷。我们可以采取如下办法来解决,修改Person类如下:

  1. package com.home;  
  2.   
  3. public class Person {  
  4.     private final Name name;  
  5.   
  6.     public Person(Name name) {  
  7.         super();  
  8.         // 设置name属性为临时创建的Name对象,该对象的firstName属性和lastName属性  
  9.         // 与传入的name对象的firstName属性和lastName属性相同  
  10.         this.name = new Name(name.getFirstName(), name.getLastName());  
  11.     }  
  12.   
  13.     public Name getName() {  
  14.   
  15.         // 返回一个匿名对象,该对象的firstName属性和lastName属性  
  16.         // 与该对象里的name属性的firstName属性和lastName属性相同  
  17.         return new Name(name.getFirstName(), name.getLastName());  
  18.     }  
  19.   
  20.     public static void main(String[] args) {  
  21.         Name n = new Name("三""张");  
  22.         Person p = new Person(n);  
  23.         System.out.println(p.getName().getFirstName());  
  24.         // 改变Person对象Name属性的firstName属性值  
  25.         n.setFirstName("无忌");  
  26.         System.out.println(p.getName().getFirstName());  
  27.     }  
  28. }  


再次运行程序,发现Person对象的Name属性的firstName属性没有改变了。

另外,由于不可变类的实例的状态不可改变,所以可以很方便地被多个对象所共享,那么如果程序要经常使用相同的不可变类实例,为了减少系统开销,一般要考虑使用缓存机制。下面使用数组作为缓存池来构建一个可以缓存实例的不可变类:

  1. package com.home;  
  2.   
  3. public class CacheImmutale {  
  4.     private final String name;  
  5.     private static CacheImmutale[] cache = new CacheImmutale[10];  
  6.     private static int pos = 0;  
  7.   
  8.     public CacheImmutale(String name) {  
  9.         super();  
  10.         this.name = name;  
  11.     }  
  12.   
  13.     public String getName() {  
  14.         return name;  
  15.     }  
  16.   
  17.     public static CacheImmutale valueOf(String name) {  
  18.         // 遍历已缓存的对象  
  19.         for (int i = 0; i < pos; i++) {  
  20.             // 如果已有相同实例,直接返回该缓存的实例  
  21.             if (cache[i] != null && cache[i].getName().equals(name)) {  
  22.                 return cache[i];  
  23.             }  
  24.         }  
  25.         // 如果缓冲池已满  
  26.         if (pos == 10) {  
  27.             // 把缓存的第一个对象覆盖  
  28.             cache[0] = new CacheImmutale(name);  
  29.             pos = 1;  
  30.             return cache[0];  
  31.         } else {  
  32.             // 把新创建的对象缓存起来,pos加1  
  33.             cache[pos++] = new CacheImmutale(name);  
  34.             return cache[pos - 1];  
  35.         }  
  36.     }  
  37.   
  38.     @Override  
  39.     public int hashCode() {  
  40.         return name.hashCode();  
  41.     }  
  42.   
  43.     @Override  
  44.     public boolean equals(Object obj) {  
  45.         if (obj instanceof CacheImmutale) {  
  46.             CacheImmutale ci = (CacheImmutale) obj;  
  47.             if (name.equals(ci.getName())) {  
  48.                 return true;  
  49.             }  
  50.         }  
  51.         return false;  
  52.     }  
  53.   
  54.     public static void main(String[] args) {  
  55.         CacheImmutale c1 = CacheImmutale.valueOf("hello");  
  56.         CacheImmutale c2 = CacheImmutale.valueOf("hello");  
  57.         System.out.println(c1 == c2);// 输出结果为true  
  58.     }  
  59. }  

对于缓存的使用,应根据系统需求而定,简单的说,如果某个对象使用的次数不多,重复使用的概率不大,就没必要使用缓存,毕竟缓存的对象也会占用系统内存。如果某个对象需要频换地重复使用,这时就应该使用缓存了。

 另外,上面的示例来源疯狂JAVA讲义一书,个人对上面那个Person类里面的属性是引用类型的解决办法存有疑问,他那种办法虽然保证的Person对象的Name属性所指对象的内容没有改变,但Person对象返回的Name属性已经不是同一个属性了,它的地址已发生改变,赋值和返回都是通过new出来的,我个人做了如下改进,觉得更合理:

  1. package com.home;  
  2.   
  3. public class Person {  
  4.     private final Name name;  
  5.   
  6.     public Person(Name name) {  
  7.         super();  
  8.         // 设置name属性为临时创建的Name对象,该对象的firstName属性和lastName属性  
  9.         // 与传入的name对象的firstName属性和lastName属性相同  
  10.         this.name = new Name(name.getFirstName(), name.getLastName());  
  11.     }  
  12.   
  13.     public Name getName() {  
  14.         // 直接返回当前实例的name属性即可  
  15.         return name;  
  16.     }  
  17.   
  18.     public static void main(String[] args) {  
  19.         Name n = new Name("三""张");  
  20.         Person p = new Person(n);  
  21.         System.out.println(p.getName() + "  " + p.getName().getFirstName());  
  22.         // 改变Person对象Name属性的firstName属性值  
  23.         n.setFirstName("无忌");  
  24.         System.out.println(p.getName() + "  " + p.getName().getFirstName());  
  25.     }  
  26. }  
从打印结果可以看出p的name属性的地址和所指内容都没变。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值