单例模式之线程安全问题

不考虑线程安全问题的时候单例模式的代码:

 1 class Bank {
 2     //把该类的构造器声明为私有
 3     private Bank() {
 4 
 5     }
 6     //设置私有化静态类变量
 7     private static Bank instance = null;
 8 
 9     //设置获取该实例的静态方法
10     public static Bank getInstance() {
11            if (instance == null) {
12                     instance = new Bank();
13                 }
14            return instance;
15     }
16 }

分析:

  单例模式通俗来说就是要求我们在创建某一个对象的时候只能创建一个这个类的对象。在多线程的情况下,可能会有多个线程在各自的run()方法中调用public static Bank getInstance()方法创建类的对象实例,如果有一个线程再调用方法且未执行完毕的时候发生阻塞,那么另外一个线程也会执行这个方法,这个时候就会有两个线程同时创建对象实例,这就违背了单例模式的初中,显然这样是线程不安全的,因此我们需要针对这个问题解决线程安全问题。

解决办法:

  在单例模式中我们可以把instance这个对象实例看成是一个共享数据,我们需要使用同步代码快或者是同步方法来解决共享数据的线程安全问题。

 1 package com.baozi.java;
 2 
 3 public class BankTest {
 4     //.......
 5 }
 6 /**
 7  * 这是一个单例模式的类:
 8  * 线程安全问题的分析:在多线程的情况下这个单例模式是非线程安全的,多线程的时候会有多个线程通过
 9  * public static Bank getInstance()方法来获取对象实例,如果某一个线程在进行判断对象实例是否已经被初始化的时候
10  *对象实例为空,紧接着会进入if()内进行创建对象实例的过程,如果在此时发生阻塞,那么下一个线程抢占到cpu执行权,
11  * 这时该线程也会进行判断且对象实例也为空同样要进去执行创建对象实例的操作,这样两个实例都要进行创建对象
12  * 实例,此时是会出现线程安全问题的。
13  *线程安全问题的解决:我们可以使用synchronized代码块或者synchronized方法来解决线程安全问题。
14  * 1、如果使用synchronized方法来解决线程安全问题:
15  *  public static synchronized Bank getInstance() {
16  *         if (instance == null) {
17  *             instance = new Bank();
18  *         }
19  *         return instance;
20  *     }
21  *   2、如果使用synchronized代码块来解决线程安全问题:
22  *    public static Bank getInstance() {
23  *         synchronized (Bank.class) {
24  *             if (instance == null) {
25  *                 instance = new Bank();
26  *             }
27  *             return instance;
28  *         }
29  *     }
30  *     或者是:这种方法效率要比上面全部放在synchronized代码块中的效率要高
31  *       public static Bank getInstance() {
32  *         if(instance == null){
33  *             synchronized (Bank.class) {
34  *                 if (instance == null) {
35  *                     instance = new Bank();
36  *                 }
37  *             }
38  *         }
39  *         return instance;
40  *     }
41  */
42 class Bank {
43     //把该类的构造器声明为私有
44     private Bank() {
45 
46     }
47     //设置私有化静态类变量
48     private static Bank instance = null;
49 
50     //设置获取该实例的静态方法
51     public static Bank getInstance() {
52         if(instance == null){
53             synchronized (Bank.class) {
54                 if (instance == null) {
55                     instance = new Bank();
56                 }
57             }
58         }
59         return instance;
60     }
61 }

 

转载于:https://www.cnblogs.com/BaoZiY/p/10727095.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
单例模式是一种常用的软件设计模式,它保证一个类只有一个实例,并提供一个全局访问点。然而,在多线程环境下,如果单例模式的实现不当,可能会导致线程安全问题。主要的线程安全问题出现在以下几个方面: 1. **静态初始化器锁**(Synchronized static block):在Java中,如果没有使用双重检查锁定(Double-Checked Locking),多个线程同时进入初始化代码块时,可能会导致初始化顺序不一致,从而创建多个实例。 ```java // 非线程安全单例 public class Singleton { private static Singleton instance; public static Singleton getInstance() { if (instance == null) { instance = new Singleton(); // 单线程环境下无问题,但在多线程下可能出问题 } return instance; } } ``` 2. **懒汉式单例**:在多线程情况下,如果只有一个同步块来创建实例,且创建实例的操作放在了if语句里,那么在多个线程首次访问时可能会看到部分初始化但未完成的实例。 ```java // 懒汉式单例(线程不安全) public class LazySingleton { private static Singleton instance; public static Singleton getInstance() { if (instance == null) { synchronized (LazySingleton.class) { if (instance == null) { instance = new Singleton(); } } } return instance; } } ``` 3. **饿汉式单例**:虽然这个版本在单线程环境中没问题,但在多线程环境中由于一开始就初始化了实例,因此会存在资源浪费和同步问题,但不会发生多个实例的问题。 ```java // 饿汉式单例线程安全,但资源消耗大) public class EagerSingleton { private static final Singleton INSTANCE = new Singleton(); public static Singleton getInstance() { return INSTANCE; } } ``` 为了解决这些问题,可以采用如下改进方案: - 使用双重检查锁定(Double-Checked Locking),保证线程安全的同时提高性能。 - 使用`enum Singleton`方式,因为Java对枚举类型的加载是原子的,所以线程安全且简单。 ```java // 双重检查锁定的单例线程安全) public class Singleton { private volatile static Singleton instance; private Singleton() {} public static Singleton getInstance() { if (instance == null) { synchronized (Singleton.class) { if (instance == null) { instance = new Singleton(); } } } return instance; } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值