【JavaEE】单例模式

饿汉模式

class Singleton {
    private static Singleton instance = new Singleton();

    public static Singleton getInstance() {
        return instance;
    }

    /**
     * private : 如果不将构造方法的访问修饰限定符修改为 private 那么就会被构造出多个实例 就违背了单例模式的初衷
     */
    private Singleton() {}
}

懒汉模式

class Singleton2 {

     private volatile static Singleton2 instance;

    public static Singleton2 getInstance() {
        if (instance == null) {
            synchronized (Singleton2.class) {
                if (instance == null) {
                    instance = new Singleton2();
                }
            }
        }
        return instance;
    }

    private Singleton2() {}

}

总结

在多线程下,饿汉模式是线程安全的,懒汉模式是线程不安全的;

在这里我们再复习一下引起线程不安全的原因:

1.多线程下,抢占式执行(操作系统的随机调度)

2.多个线程修改同一个变量

3.修改操作不是原子的

4.内存可见性

5.指令重排序

public static Singleton2 getInstance() {
    if (instance == null) {
        instance = new Singleton2();
    }
    return instance;
}

懒汉模式:

当一个线程t1正在创建对象并赋值给instance,但并没有执行完时,另一个线程t2读到了还没有修改完的值,因此又重新创建了一个新的对象,此时就会导致实例被创建出多份儿了。

那么如何解决呢? - 加锁

public static Singleton2 getInstance() {
    synchronized (Singleton2.class) {
        if (instance == null) {
            instance = new Singleton2();
        }
    }
    return instance;
}

此时就不会出现上述问题了,但是又有新的问题出现了

仔细观察我们可以发现,我们只需要在第一次进行加锁操作,当时里被创建后便不需要了,直接返回即可,如果每次都需要加锁,那岂不是很烦。因此我们需要在进入加锁操作前再加上一层if语句,来判断一下是否需要创建实例。

public static Singleton2 getInstance() {
    if (instance == null) {
        synchronized (Singleton2.class) {
            if (instance == null) {
                instance = new Singleton2();
            }
        }
    }
    return instance;
}

上述代码,还有一个非常重要的问题!

假设两个线程t1,t2调用getInstance方法,t1拿到锁,进入new操作,这里的new操作可以粗略的分成三个步骤:

1.申请内存,得到内存首地址

2.调用构造方法,初始化实例

3.将内存首地址赋值给instance

上述过程编译器可能会进行“指令重排序”的优化

如果是单线程,指令2和3发生调换是无所谓的

如果是多线程,就会出现问题。

假设t1线程拿到锁正在进行 new 操作 ,且发生指令重排序,先执行指令3,此时线程t2判断instance不为空,直接返回instance,那么我们知道,此时t1线程虽然把内存首地址赋值给了instance,但里面的数据是无效的,而t2线程拿到instance后,可能会对其进行解引用操作(使用里面的属性/方法),就会出现问题!

那么如何防止呢?也是需要用到 volatile 关键字 - 防止指令重排序

private volatile static Singleton2 instance;

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
您好!对于JavaEE开发,Eclipse是一个非常流行的集成开发环境(IDE)。它为JavaEE开发者提供了丰富的功能和工具,使他们能够更轻松地构建和管理JavaEE应用程序。 在Eclipse中进行JavaEE开发,您可以使用Eclipse的JavaEE插件(通常称为Eclipse Enterprise Edition for JavaEE),它包含了一系列用于JavaEE开发的工具和模块。这些工具包括Servlet容器、JSP编辑器、EJB开发工具、数据库连接工具等等。 使用Eclipse进行JavaEE开发的步骤如下: 1. 安装Eclipse:您可以从Eclipse官方网站下载适合您操作系统的Eclipse版本,并按照安装指南进行安装。 2. 安装JavaEE插件:在Eclipse中,通过插件安装向导安装JavaEE插件。您可以在Eclipse的市场中搜索并安装适合您版本的JavaEE插件。 3. 创建JavaEE项目:在Eclipse中,可以通过菜单或快捷键创建新的JavaEE项目。选择适当的项目类型(如Dynamic Web Project、EJB Project等),并按照向导进行项目配置。 4. 开发和调试代码:在Eclipse中,您可以使用内置的编辑器编写JavaEE代码,使用调试器进行调试,并利用各种工具和视图来构建和测试您的应用程序。 5. 部署和运行应用程序:完成代码的开发和调试后,您可以将您的JavaEE应用程序部署到适当的服务器上,并启动服务器以运行您的应用程序。 当然,这只是一个简要的介绍。对于JavaEE开发,Eclipse还提供了许多其他功能和工具,如代码自动完成、版本控制、构建工具集成等等。 希望这能帮助到您!如果您还有其他问题,请随时提问。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值