今天看到群里,关于单例模式的多线程下的安全问题:
1:最简单写法的线程安全的单例,但是直接是在内存的,占空间
package com.http.concurrent.test; public class Singleton { private static Singleton instance=new Singleton(); // 直接new 可以加上final private Singleton() { // 私有 } public static Singleton getInstance() { return instance; } }
2:线程安全下 有问题的:
package com.http.concurrent.test; public class Singleton { private static Singleton instance; private Singleton() { } public static Singleton getInstance() { // 1 if(instance == null) { // 2 instance=new Singleton(); // 3 } return instance; } }
如果线程B,C,同时访问到了2,;A到3处,new,此时 明显B,C会再次new; 产生多个实例
3:简单的线程安全
public class Singleton { private static Singleton instance; private Singleton() { } public static synchronized Singleton getInstance() { // 1 if(instance == null) { // 2 instance=new Singleton(); // 3 } return instance; } }
synchronized 锁的范围太大,效率比较低。
4:缩小synchronized的锁的范围,所称double-check
package com.http.concurrent.test; public class Singleton { private static Singleton instance; private Singleton() { } public static Singleton getInstance() { // 1 if(instance == null) { // ** synchronized(Singleton.class) { if(instance == null) { // 2 instance=new Singleton(); // 3 } } } return instance; } }
2必须有,不然并发情况下 ,可能会new 多Sinleton;
为了提高效率可以在上面的double-check修改下:
private volatile static Singleton instance;
加上volatile后instance变量就对 所有的线程可见了,有其他线程new一个实例,立马就能看见,**处就可以发现
/** * 静态内部类 单例模式 * * @author ljq * @version 1.0 * @since 1.0 */ public class Singleton { private Singleton() { } public static final Singleton getInstance() { return SingletonHolder.instance; } /** * 在第一次被引用时被加载 * 由于内部静态类只会被加载一次,故该实现方式时线程安全的 */ private static class SingletonHolder { protected static final Singleton instance = new Singleton(); } public static void main(String[] args) { Singleton s = Singleton.getInstance(); Singleton s1 = Singleton.getInstance(); System.out.println(s == s1); } }
并发容器实现的单例模式:
package xylz.study.concurrency; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentMap; public class ConcurrentDemo1 { private static final ConcurrentMap<String, ConcurrentDemo1> map = new ConcurrentHashMap<String, ConcurrentDemo1>(); private static ConcurrentDemo1 instance; public static ConcurrentDemo1 getInstance() { if (instance == null) { map.putIfAbsent("INSTANCE", new ConcurrentDemo1()); instance = map.get("INSTANCE"); } return instance; } private ConcurrentDemo1() { } }
当然这里只是一个操作的例子,实际上在单例模式文章中有很多的实现和比较。清单2 在存在大量单例的情况下可能有用,实际情况下很少用于单例模式。但是这个方法避免了向Map中的同一个Key提交多个结果的可能,有时候在去掉重复记录上很有用(如果记录的格式比较固定的话)。
相关推荐
单例模式是一种设计模式,旨在确保一个类只有一个实例,并提供全局访问点。在单例模式中,类的构造函数是私有的,防止外部直接创建对象,而是通过静态方法获取该类的唯一实例。单例模式的唯一性通常是在进程范围内,...
"设计模式单例模式和工厂模式综合应用"的主题聚焦于两种常用的设计模式:单例模式和工厂模式,并探讨它们如何协同工作来实现高效、灵活的代码结构。这个主题尤其适用于Java编程语言,因为Java的面向对象特性使得设计...
单例模式是软件设计模式中的一种经典模式,它保证了类只有一个实例存在,并提供一个全局访问点。在Java等面向对象编程语言中,单例模式常用于管理共享资源,如数据库连接池、线程池或者配置文件等。结合工厂模式,...
单例模式是软件设计模式中的一种经典模式,用于确保一个类只有一个实例,并提供一个全局访问点。在Java中,有多种实现单例模式的方法,每种都有其特点和适用场景。接下来,我们将深入探讨这些实现方式。 首先,我们...
在C++编程中,单例模式是一种常用的软件设计模式,它保证一个类只有一个实例,并提供一个全局访问点。线程安全的单例模式在多线程环境下尤其重要,因为不正确的实现可能导致多个线程创建多个实例,这违反了单例模式...
**设计模式——单例模式** 在软件工程中,设计模式是一种在特定场景下解决常见问题的标准方案,可以被复用并提升代码质量。单例模式是设计模式中的一种,它保证一个类只有一个实例,并提供一个全局访问点。这种模式...
单例模式是软件设计模式中的一种,它保证一个类只有一个实例,并提供一个全局访问点。在C#中,单例模式常用于管理共享资源或控制类的实例化过程,以提高性能、节约系统资源,特别是在整个应用程序生命周期内只需要一...
单例模式是软件设计模式中的一种经典模式,其主要目的是确保一个类只有一个实例,并提供一个全局访问点。这种模式在很多场景下非常有用,比如控制共享资源、管理配置对象等。下面将详细介绍七种常见的单例模式实现...
单例模式是软件设计模式中的一种,它的主要目的是确保一个类只有一个实例,并提供一个全局访问点。在Java或类似的面向对象编程语言中,单例模式常用于管理共享资源,如数据库连接池、线程池或者配置文件等。在这个...
在C++编程中,单例模式是一种常用的软件设计模式,它保证一个类只有一个实例,并提供一个全局访问点。在这个特定的场景中,我们讨论的是一个实现了单例模式的日志类,该类专为多线程环境设计,具备日志等级控制、...
首先向关注过我这个系列...这立刻让我想到了最常用也是最简单最容易理解的一个设计模式 单例模式 何为 单例模式 ? 故名思议 即 让 类 永远都只能有一个实例。 由于 示例代码 比较简单 我也加了注释,这里就不在赘述
单例模式是软件设计模式中的一种,它的主要目的是确保一个类只有一个实例,并提供一个全局访问点。这种模式在很多场景下都非常有用,比如控制资源的唯一性、管理共享配置或者创建昂贵的对象时避免频繁创建销毁。 ...
Java中的单例模式是一种常用的软件设计模式,它保证一个类只有一个实例,并提供全局访问点。在Java编程中,单例模式常用于控制资源的访问,比如数据库连接池、线程池或者日志对象等。本篇文章将深入探讨如何在Java中...
单例模式是一种设计模式,它的主要目标是确保一个类只有一个实例,并提供一个全局访问点。在软件工程中,单例模式常用于控制资源的共享,比如数据库连接池、线程池或者日志系统等,这些资源通常需要全局唯一且高效地...
单例模式是软件设计模式中的一种,它的核心思想是确保一个类在整个系统中只有一个实例,并提供一个全局访问点。在Java或类似编程语言中,单例模式常常被用来管理资源,比如数据库连接、线程池或者配置信息,因为这些...
在Qt的Qml环境中,单例模式是一种设计模式,它允许在整个应用程序中创建一个全局访问点,确保某个类只有一个实例存在。这样的设计模式在需要共享数据或者服务时非常有用,避免了多处创建相同对象导致的数据不一致或...
单例模式是软件设计模式中的经典模式之一,其主要目的是控制类的实例化过程,确保在应用程序的整个生命周期中,某个类只有一个实例存在。这样的设计通常适用于那些需要频繁创建和销毁,但资源消耗较大的对象,如...
单例模式是一种常用的设计模式,它的核心思想是在整个应用程序中,一个类只能有一个实例存在。单例模式常用于控制资源的共享,例如数据库连接池、日志服务等。单例模式有多种实现方式,常见的包括懒汉式、饿汉式以及...
单例模式是软件设计模式中的一种,它保证一个类只有一个实例,并提供一个全局访问点。在C++中,实现单例模式有多种方法,我们将会深入探讨这一模式的原理、优缺点以及如何在实际编程中应用。 单例模式的核心在于...