解决多线程环境下单例模式同时访问生成多个实例

如何满足单例:1.构造方法是private、static方法、if语句判断

①、单线程

Single类

复制代码
//Single类,定义一个GetInstance操作,允许客户访问它的唯一实例。GetInstance是一个静态方法,主要负责创建自己的唯一实例
public class LazySingleton {
    private static LazySingleton instance;

    private LazySingleton() {
        System.out.println("创建一次");
    }

    public static LazySingleton GetInstance() {
        //当多线程来临的时候判断是否为null,此时instance就是临界资源,会实例化多个
        if (instance == null) {
            instance = new LazySingleton();
        }
        return instance;
    }
}
复制代码
//客户端代码
java 复制代码
public class Main {
    public static void main(String[] args) {
            LazySingleton s1= LazySingleton.GetInstance();
            LazySingleton s2=  LazySingleton.GetInstance();

            if(s1==s2){
                System.out.println("两个对象是相同的实例");
            }
    }
}

运行结果:

这样的话就满足了单例的效果,保证只实例化一个类,**因为LazySingleton封装它的唯一实例,这样它可以严格地控制客户怎样访问它以及何时访问它。简单地说就是对唯一实例的受控访问。**客户端通过那唯一可以访问的GetInstance方法来访问那一个实例。但如果是多个线程同时调用GetInstance方法,同时运行到了GetInstance方法这儿,它们都会去判断有没有被实例化,判断都为True,那样的话就创建了两个实例,就违背了单例模式,不是一个单例。看下多线程下的单例:

②、多线程

单例类

java 复制代码
public class LazySingleton {
    private static LazySingleton instance;

    private LazySingleton() {
        System.out.println("创建一次");
    }

    public static LazySingleton GetInstance() {
        //当多线程来临的时候判断是否为null,此时instance就是临界资源,会实例化多个
        if (instance == null) {
            instance = new LazySingleton();
        }
        return instance;
    }
}

main函数

java 复制代码
public class Main {
    public static void main(String[] args) {
        Runnable r=()->{

            LazySingleton s1= LazySingleton.GetInstance();
            LazySingleton s2=  LazySingleton.GetInstance();

            if(s1==s2){
                System.out.println("两个对象是相同的实例");
            }
        };

        //两个线程
        Thread t1= new Thread(r);
        Thread t2= new Thread(r);

        t1.start();
        t2.start();
    }
}

运行结果:

我们会发现对象被创建了两次,我们通过调试发现s1和s2两个对象的地址实际上是不一样的:

当线程t1刚执行完if (instance == null)判断之后时间片到了,t2线程执行完if (instance == null)判断之后就进入方法体生成了实例,此时t1线程又获得了时间片,t1会接着上次中断的地方继续执行,t1线程便会进入方法体又生成了一个新的实例,此时t1和t2线程各生成了一个实例

如何解决这样一个问题呢?

添加锁,当线程位于临界区的时候就上锁,其他线程来临的时候只能在外排队等待。

③、多线程单例------单锁

单例类

java 复制代码
package com.example;

public class LazySingleton {
    private static LazySingleton instance;

    private LazySingleton() {
        System.out.println("创建一次");
    }

    public static LazySingleton GetInstance() {
        //当多线程来临的时候判断是否为null,此时instance就是临界资源,会实例化多个
        //方法:加锁-把判断的这部分逻辑上锁
        synchronized ("") {
            if (instance == null) {
                instance = new LazySingleton();
            }
        }
        return instance;
    }

}

main函数

java 复制代码
public class Main {
    public static void main(String[] args) {
        Runnable r=()->{

            LazySingleton s1= LazySingleton.GetInstance();
            LazySingleton s2=  LazySingleton.GetInstance();

            if(s1==s2){
                System.out.println("两个对象是相同的实例");
            }
        };

        //两个线程
        Thread t1= new Thread(r);
        Thread t2= new Thread(r);

        t1.start();
        t2.start();
    }
}

运行结果:

发现对象创建了一次。在同一时刻加了锁的那部分程序只有一个线程可以进入,我们可以让最先进入的那个线程先上一把锁,创建实例。后面在进入的线程就不会再去创建对象实例了,因为第一名来的线程已经创建了,你这个判断的结果是False,自然无法创建了。这样的话就保证了多个线程同时访问的话不会有多个实例化。解决了上面单实例带来的问题。但每次进入的线程都需要先加锁在判断,我都还不知道有没有创建过这个实例呢你就让我加锁,第一名已经实例化过了,我进去再加锁,在判断一次,如果有上百个线程同时访问呢,这样的工作重复上百次,不是很影响我这个程序的性能吗?我们就可以用到双重锁定来解决这个问题

④、多线程------双重锁(Double-Check Locking)

java 复制代码
package com.example;

public class DoubleLockSingleton {
    private static DoubleLockSingleton instance;

    private DoubleLockSingleton() {
        System.out.println("实例化了一次");
    }

    public static DoubleLockSingleton GetInstance() {
        //第一层判断:先判断实例是否存在,不存在再加锁处理
        if (instance == null) {
            synchronized ("") {
                //第二层判断
                if (instance == null) {
                    instance = new DoubleLockSingleton();
                }
            }
        }
        return instance;
    }
}

通过这样两重的判断,进入的线程不用每次都加锁,只是在实例未被创建的时候在加锁处理。同时也保证多线程的安全。

相关推荐
唐青枫3 分钟前
Java JdbcTemplate 实战指南:用 Spring 轻量完成数据库增删改查
java·spring boot·spring
未秃头的程序猿5 分钟前
别再让大模型单打独斗了!Java 多 Agent 协作实战:任务拆解+结果聚合
java·后端·ai编程
右耳朵猫AI7 分钟前
Java & JVM技术周刊 2026年第20周
java·开发语言·jvm
三雒7 分钟前
KMP 实战:Android 开发如何快速统一双端 IM 模块
android·ios·kotlin
人道领域8 分钟前
【LeetCode刷题日记】538.把二叉搜索树转换为累加树
java·开发语言·后端·算法·leetcode
铁皮哥9 分钟前
【后端开发】什么是守护线程,和普通线程有什么区别?
java·开发语言·数据库·人工智能·python·spring·intellij-idea
西凉的悲伤11 分钟前
Spring Boot + ShardingSphere 介绍
java·spring boot·后端·shardingsphere·分库分表
Lsk_Smion15 分钟前
力扣实训 _ [33].搜索旋转排序数组 _ [92].翻转链表Ⅱ
java·数据结构·算法
AI人工智能+电脑小能手17 分钟前
【大白话说Java面试题 第86题】【Mysql篇】第16题:MySQL 中锁的种类与行锁实现原理?
java·开发语言·数据库·mysql·面试
日月云棠23 分钟前
12 Enum —— 枚举类型的底层实现
java·后端