JAVA进阶 JAVA中具有实例缓存的不可变类

4/2/2008来源:Java教程人气:5248

 

不可变类的实例的状态不会变化,这样的实例可以安全地被其他与之关联的对象共享,还可以安全地被多个线程共享。为了节省内存空间,优化程序的性能,应该尽可能地重用不可变类的实例,避免重复创建具有相同属性值的不可变类的实例。

在JDK 1.5的基本类库中,对一些不可变类,如Integer类做了优化,它具有一个实例缓存,用来存放程序中经常使用的Integer实例。JDK 1.5的Integer类新增了一个参数,为int类型的静态工厂方法valueOf(int i),它的处理流程如下:

if(在实例缓存中存在取值为i的实例)
   直接返回这个实例
else{
  用new语句创建一个取值为i的Integer实例
  把这个实例存放在实例缓存中
  返回这个实例
}

在以下程序代码中,分别用new语句和Integer类的valueOf(int i)方法来获得Integer实例。

Integer a=new Integer(10);
Integer b=new Integer(10);

Integer c=Integer.valueOf(10);
Integer d= Integer.valueOf(10);

System.out.PRintln(a==b); //打印false
System.out.println(a==c); //打印false

System.out.println(c==d); //打印true

       以上代码共创建了3个Integer对象,每个new语句都会创建一个新的Integer对象。而Integer.valueOf(10)方法仅在第一次被调用时,创建取值为10的Integer对象,在第二次被调用时,直接从实例缓存中获得它。由此可见,在程序中用valueOf()静态工厂方法获得Integer对象,可以提高Integer对象的可重用性。

     到底如何实现实例的缓存呢?缓存并没有固定的实现方式,完善的缓存实现不仅要考虑何时把实例加入缓存,还要考虑何时把不再使用的实例从缓存中及时清除,以保证有效合理地利用内存空间。一种简单的实现是直接用java集合来作为实例缓存。

     下面的例程,它拥有实例缓存和相应的静态工厂方法valueOf()。Name类的实例缓存中可能会加入大量Name对象,为了防止耗尽内存,在实例缓存中存放的是Name对象的软引用(SoftReference)。如果一个对象仅仅持有软引用,Java虚拟机会在内存不足的情况下回收它的内存。

例程11-12 Name.java

import java.util.Set;
import java.util.HashSet;
import java.util.Iterator;
import java.lang.ref.*;

public class Name {

//实例缓存,存放Name对象的软引用

private static final Set>   names=new HashSet>();

public static Name valueOf(String firstname, String lastname){ //静态工厂方法

 Iterator> it=names.iterator();

 while(it.hasNext()){
  SoftReference  ref=it.next();//获得软引用
  Name name=ref.get();//获得软引用所引用的Name对象
 if(name!=null&& name.firstname.equals(firstname)&& name.lastname.equals(lastname))
   return name;
 }

 //如果在缓存中不存在Name对象,就创建该对象,并把它的软引用加入到实例缓存

 Name name=new Name(firstname,lastname);
 names.add(new SoftReference(name));
 return name;

}

public static void main(String args[]){

  Name n1=Name.valueOf("小红","王");
  Name n2=Name.valueOf("小红","王");
  Name n3=Name.valueOf("小东","张");
  System.out.println(n1);
  System.out.println(n2);
  System.out.println(n3);
   System.out.println(n1==n2); //打印true
 }

 }

在程序中,既可以通过new语句创建Name实例,也可以通过valueOf()方法创建Name实例。在程序的生命周期中,对于程序不需要经常访问的Name实例,应该使用new语句创建它,使它能及时结束生命周期;对于程序需要经常访问的Name实例,那就用valueOf()方法来获得它,因为该方法能把Name实例放到缓存中,使它可以被重用。