您的位置:首页 > 编程语言 > Java开发

Java设计模式之单例

2016-05-27 15:03 337 查看
一、Java中的单例:

特点:

① 单例类只有一个实例

② 单例类必须自己创建自己唯一实例

③ 单例类必须给所有其他对象提供这一实例

二、两种模式:

①懒汉式单例<线程不安全>

在类加载时,不创建实例,运行调用时创建。类加载快,在运行时获取对象速度慢

示例:

//懒汉模式
public class Pet {

private Pet(){

}

private static Pet pet=null;

public static Pet getInfo(){
if(pet==null){
pet=new Pet();
}
return pet;
}

}


②饿汉式单例<线程安全>

在类加载的时候,就完成初始化。所以类加载慢,但是在运行时获取对象快

示例:

//饿汉模式
public class Pet {

private Pet(){

}

private static Pet pet=new Pet();

public static Pet getInfo(){
return pet;
}

}


饿汉模式线程安全,但是,懒汉模式线程安全性不高,若是多线程,又怎能保证单例?

第一种方法:同步

在方法调用上加了同步,虽然线程安全了,但是每次都要同步,会影响性能,毕竟99%的情况下是不需要同步的

public static synchronized Pet getInfo(){
if(pet==null){
pet=new Pet();
}
return pet;
}


第二种方法:双重检查锁定

在getInfo中做了两次null检查,确保了只有第一次调用单例的时候才会做同步,这样也是线程安全的,同时避免了每次都同步的性能损耗

所谓“双重检查加锁”机制,指的是:并不是每次进入getInstance方法都需要同步,而是先不同步,进入方法后,先检查实例是否存在,如果不存在才进行下面的同步块,这是第一重检查,进入同步块过后,再次检查实例是否存在,如果不存在,就在同步的情况下创建一个实例,这是第二重检查。这样一来,就只需要同步一次了,从而减少了多次在同步情况下进行判断所浪费的时间。

  “双重检查加锁”机制的实现会使用关键字volatile,它的意思是:被volatile修饰的变量的值,将不会被本地线程缓存,所有对该变量的读写都是直接操作共享内存,从而确保多个线程能正确的处理该变量。

public static Pet getInfo(){
if(pet==null){
synchronized(Pet.class){
if(pet==null){
pet=new Pet();
}
}
}
return pet;
}


第三种方法:静态内部类

利用了classloader的机制来保证初始化时只有一个线程,所以也是线程安全的,同时没有性能损耗

public class Pet {
private static class LazyHolder {
private static final Pet INSTANCE = new Pet();
}
private Pet (){}
public static final Singleton getInfo() {
return LazyHolder.INSTANCE;
}
}


资源加载和性能:

饿汉模式在类创建的同时就实例化一个静态对象,不管之后会不会使用这个单例,都会占用一定的内存,但是相应的,在第一次调用时速度也会非常快,因为其资源已经初始化完成。

懒汉模式会延迟加载,在第一次使用该单例的时候才会实例化对象出来,第一次调用时要做初始化,如果要做的工作比较多,性能上会有些延迟,之后就和饿汉模式一样可。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: