单例模式 -- 懒汉模式&饿汉模式

目录

一、单例模式是什么?

二、饿汉模式

三、懒汉模式

一、单例模式是什么?

单例模式是一种设计模式,用于将类的实例化限制为一个对象。它确保一个类只有一个实例,并提供了该实例的全局访问点。这种模式被广泛用于创建对象的唯一实例,例如数据库连接和日志记录器等。单例模式可以在多线程环境下使用,确保多个线程访问同一个实例时不会发生冲突。常见的实现方式包括懒加载、饿汉式和双重检查等。

二、饿汉模式

饿汉模式是一种单例模式的实现方式,它的特点是在程序启动时就进行对象的实例化。在这种实现方式中,单例对象会在类加载时就被创建出来,因此也被称作“饱汉模式”或者“静态常量方式”。

饿汉模式的优点是实现简单、线程安全,无需考虑多线程环境下的同步问题。其缺点是无法进行懒加载,带来了一定的系统资源开销。在某些场景下,如果单例对象很大或者初始化耗时较长,饿汉模式的开销可能会比较明显。

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

三、懒汉模式

懒汉模式是一种单例模式的实现方式,它的特点是在首次访问单例对象时才进行对象的实例化。懒汉模式的优点是可以避免在程序启动时就进行对象的实例化,节省了系统的资源。懒汉模式的缺点是在多线程环境下,可能会出现线程安全问题,需要额外的同步措施来保证线程安全性。

例如,以下是一个简单的懒汉模式实现:

class Singleton{
    private static Singleton singleton = null;
    public Singleton getSingleton() {
        if (singleton==null){
            this.singleton = new Singleton();
        }
        return singleton;
    }
    private Singleton(){}
}
上面的懒汉模式的实现是线程不安全的.
线程安全问题发生在首次创建实例时. 在多个线程中同时调用 getSingleton 方法,可能两个线程会同时进行new操作 , 这样我们多线程创建了多个对象 , 就不是单例模式了 , 主要导致该问题的是由于这里面涉及到了对singleton的修改操作,失去了原子性,为了保证原子性,我们想到了加锁,从而实现线程安全问题。
我们为了解决这个多线程竞争的安全问题我们可以进行加锁操作 如下面代码
class Singleton{
    private static Singleton singleton = null;
    synchronized public Singleton getSingleton() {
        if (singleton==null){
            this.singleton = new Singleton();
        }
        return singleton;
    }
    private Singleton(){}
}

但是现在在我们进行 new 操作的时候可能会触发指令重排序 , 所以我们为了避免这个情况还需要对代码进行修改

class Singleton{
    volatile private static Singleton singleton = null;
     public Singleton getSingleton() {
         if (singleton==null){
             synchronized(Singleton.class){
                 if (singleton==null){
                     this.singleton = new Singleton();
                 }
             }
         }
         return singleton;
     }
    private Singleton(){}
}
  • 这里我们使用双if减少不必要的加锁操作
  • 加锁是为了把if和变成原子操作
  • 使用volatile禁止指令重排序
  • 7
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
Java单例模式包括饿汉式和懒汉式两种实现方式。饿汉式是在类加载阶段就创建实例并持有,而懒汉式则是在需要时才创建实例。 饿汉模式是指在类加载阶段就创建出实例的,因此它的实例化过程相对于普通情况要早很多。这也是为什么叫“饿汉”的原因,就像一个饥饿的人对食物没有抵抗力,一下子就开始吃了一样。 懒汉模式是指在需要时才创建实例。这种方式的优点是节省了资源,只有在需要时才会进行实例化。但是它的缺点是在多线程环境下可能会导致多个线程同时创建实例的问题,需要进行额外的线程安全措施来解决这个问题。 总结来说,饿汉式适合在应用启动时就需要创建实例的情况,因为它的实例化过程早于普通情况。而懒汉式适合在需要时才创建实例的情况,可以节省资源。 需要注意的是,单例模式的使用要根据具体的适应场景来决定,不同的情况下选择不同的实现方式。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [Java设计模式单例模式——饿汉式、懒汉式(初了解)](https://blog.csdn.net/m0_68062837/article/details/127307310)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [Java多线程案例之单例模式饿汉懒汉)](https://blog.csdn.net/qq_63218110/article/details/128738155)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值