设计模式(1)-单例模式

定义:
单例模式,是一种常用的软件设计模式。在它的核心结构中只包含一个被称为单例的特殊类。通过单例模式可以保证系统中一个类只有一个实例。即一个类只有一个对象实例。

特点

  1. 单例类只能有一个实例。
  2. 单例类必须自己自己创建自己的唯一实例。
  3. 单例类必须给所有其他对象提供这一实例

单例模式构造要点

  • 私有的构造方法
  • 指向自己实例的私有静态引用
  • 以自己实例为返回值的静态的共有方法

    饿汉式

public class Singleton {  
    private static Singleton singleton = new Singleton();  
    private Singleton(){}  
    public static Singleton getInstance(){  
        return singleton;  
    }  
}

懒汉式

public class Singleton {  
    private static Singleton singleton;  
    private Singleton(){}  
    //加锁保持同步
    public static synchronized Singleton getInstance(){  
        if(singleton==null){  
            singleton = new Singleton();  
        }  
        return singleton;  
    }  
}  

上面的方式考虑到线程安全的问题,线程不安全时会出现多个Singleton实例,

双重锁的形式

public class Singleton{    
    private static volatile Singleton instance=null;    
    private Singleton(){        
    //TO DO
    }    
    public static  Singleton getInstance(){        
        if(instance==null){            
            synchronized(SingletonClass.class){                
                if(instance==null){
                    instance=new Singleton();
                }
            }
        }        
    return instance;
     }
}

这个模式将同步内容下方到if内部,提高了执行的效率,不必每次获取对象时都进行同步,只有第一次才同步,创建了以后就没必要了。
这种模式中双重判断加同步的方式,比第一个例子中的效率大大提升,因为如果单层if判断,在服务器允许的情况下,假设有一百个线程,耗费的时间为100*(同步判断时间+if判断时间),而如果双重if判断,100的线程可以同时if判断,理论消耗的时间只有一个if判断的时间。
所以如果面对高并发的情况,而且采用的是懒汉模式,最好的选择就是双重判断加同步的方式。
单例模式的优缺点
优点:

1.在内存中只有一个对象,节省内存空间。
2. 避免频繁的创建销毁对象,可以提高性能。
3.避免对共享资源的多重占用。
4.可以全局访问。
缺点:
1.扩展困难,由于getInstance静态函数没有办法生成子类的实例。如果要拓展,只有重写那个类
2. 隐式使用引起类结构不清晰。
3. 导致程序内存泄露的问题。

适用场景:
由于单例模式的以上优点,所以是编程中用的比较多的一种设计模式。以下为使用单例模式的场景:
1,需要频繁实例化然后销毁的对象。
2,创建对象时耗时过多或者耗资源过多,但又经常用到的对象。
3,资源共享的情况下,避免由于资源操作时导致的性能或损耗等
4,控制资源的情况下,方便资源之间的互相通信。

单例模式注意事项:
只能使用单例类提供的方法得到单例对象,不要使用反射,否则将会实例化一个新对象。
不要做断开单例类对象与类中静态引用的危险操作。
多线程使用单例使用共享资源时,注意线程安全问题。
Java中单例模式的一些常见问题
1.单例模式的对象长时间不用会被jvm垃圾收集器收集吗
人为的断开静态引用到单例对象的连接,否则jvm虚拟机不会回收单例对象

jvm回收类的判定条件:
1.该类的所有实例都已经回收,也就是java堆中不存在该类的任何实例
2.加载该类的classLoader已经回收
3.该类对应的java.lang.Class对象没有任何地方被引用,无法在任何地方通过反射访问该类的方法
满足上述三个条件jvm才会垃圾收集是回收类

2.在jvm中会出现多个单例对象吗?
在同一个jvm中使用单例模式中只能得到同一个单例,除非使用反射将可得到新的实例

try {
           Class c=Class.forName(Singleton.class.getName());
            try {
                Constructor ct=c.getDeclaredConstructor();
                ct.setAccessible(true);
                Singleton singleton=(Singleton)ct.newInstance();
                toastShort("singleton的实例通过反射获取的");
            } catch (NoSuchMethodException e) {
                e.printStackTrace();
            } catch (IllegalAccessException e) {
                e.printStackTrace();
            } catch (InstantiationException e) {
                e.printStackTrace();
            } catch (InvocationTargetException e) {
                e.printStackTrace();
            }
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值