双重检测单例类

单例类有很多种,有饿汉式,懒汉式。其中懒汉式由于其两次判断被称为双重检测单例类。

看一段代码。

 1 public class StoreKeeper {
 2     /** 属性列表值. */
 3     private HashMap<String, Store> storepool = null;
 4     private HashMap<String, Long> lifepool = null;
 5     private static StoreKeeper instance;
 6     private StoreKeeper(){
 7         storepool = new HashMap<String, Store>();
 8         lifepool = new HashMap<String, Long>();
 9     }
10     public static StoreKeeper getInstance(){
11         if (instance == null) {
12             synchronized (StoreKeeper.class) {
13                 if (instance == null) {
14                     instance = new StoreKeeper();
15                 }
16             }
17         }
18         return instance;
19     }

双重检测单例类即不会每次调用都要获取锁,又做到了延迟加载所以大家都喜欢使用到这种方式,但是双重检测模式在java里面有些问题,主要是JVM会对代码执行重排序等其他的优化策略,所以导致可能返回的实例对象中的成员变量还没有初始化,导致其他线程引用这个对象时得到的成员变量数据不一致,因为获取到的这个实例中的成员变量还未初始化为用户指定的值

解决方法:

在实例变量增加关键字volatile

private static volatile StoreKeeper instance;

这样就不会出现问题了,因为jvm看到有这个volatile关键字就不会对这段代码

进行重排序优化,所以双重检测机制都要求加上这个volatile关键字.

时间: 2024-10-17 21:27:49

双重检测单例类的相关文章

C#单例类的实现

C#单例类的实现 单例类保证一个类全局仅有一个实例,并提供一个全局访问点,由于只能生成一个实例,因此我们必须把构造函数设为私有函数以禁止他人创建实例. 实现1:懒汉式,线程不安全 该实现没有额外开销,不要求线程安全的情况下可以使用: public class Singleton1 { private static Singleton1 instance = null; private Singleton1() { } public static Singleton1 Instance { get

java单例类/

java单例类  一个类只能创建一个实例,那么这个类就是一个单例类 可以重写toString方法 输出想要输出的内容 可以重写equcal来比较想要比较的内容是否相等 对于final修饰的成员变量 一但有了初始值,就不能被重新赋值 static修饰的成员变量可以在静态代码块中 或申明该成员时指定初始值 实例成员可以在非静态代码块中,申明属性,或构造器中指定初始值 final修饰的变量必须要显示初始化 final修饰引用变量不能被重新赋值但是可以改变引用对象的内容分(只要地址值不变) final修

c++:自己动手实现线程安全的c++单例类

前段时间使用c++做项目开发,需要根据根据配置文件路径加载全局配置文件,并对外提供唯一访问点.面对这样一个需求,自然的就想到了使用单例模式来创建一个单例配置对象,供外部调用.一开始想使用boost中自带的单例类来实现,但是遗憾的是,boost中的的单例类好像只能使用无参的类构造函数,而我希望将配置文件路径作为单例配置对象的构造函数参数,此外正好借此机会使用c++自己动手实现一个单例类. 1.线程安全的c++单例类 实现线程安全的c++单例类,主要要实现以下几点:1)构造函数私有化,即构造函数.拷

创建一个单例类

关于单例的概念此处不做表述,直接上代码演示如何创建一个单例类. 1 #import <Foundation/Foundation.h> 2 3 @interface MJDemo : NSObject 4 5 + (instancetype)sharedDemo; 6 7 @end 8 9 10 #import "MJDemo.h" 11 12 @implementation MJDemo 13 14 //在iOS中所有对象分配内存空间,最终都会调用allocWithZon

单例类的总结

/** * 单例类的两种方法. * 1.饿汉式 * 当类一加载就开始创建对象 * 2.懒汉式 * 当需要使用到类的对象是调用类的类方法才开始创建对象,想比较上面而言是 * 懒汉式. * 注意,单例类的特点: * 1.必须要将构造函数私有化,这样才能防止别的类通过构造方法来建立对象,但是也是因为如此,所以 * 单例类的成员变量和成员函数都必须要用static修饰,因此要想调用单例类的函数,只能通过类名.类方法的方式来进行 * 调用. * 2.必须要在类内创建一个本类的对象. * 3.向外提供一个公

第62课 单例类模板

1. 单例需求的提出 (1)在架构设计时,某些类在整个系统生命期中最多只能有一个对象存在(Single Instance) (2)问题:如何定义一个类,使得这个类最多只能创建一个对象 2. 单例模式 (1)要控制类的对象数目,必须对外隐藏构造函数 (2)思路 ①将构造函数的访问属性设置为private ②定义instance,并初始化为NULL ③提供静态函数getInstance来返回单例实例,并判断当instance的值为空时,并创建对象,并保存在instance指针中,非空值时直接返回in

单例类的实现

单例类:大概理解为只有一个对象的类,无论怎么创建都只有会分配一个内存 这几天,需要实现某个功能,而且在很多的类里面都要调用这个方法,所以我索兴就封装了一个类,直接调用这个类里面的方法就可以实现了,不需要每次都去写同一个方法(这是大背景),之后在在调用的时候我又发现我需要每次都要用同一个对象(重点来了),所以我才需要封装一个单例的类,而且这样不会浪费内存. 好了,直接上代码了: +(UIView*)initActivityIndictionView { static UIView *view =

单例类的创建

#import "SingalTon.h" @implementation SingalTon //实例变量不能在静态方法中使用 //需要定义成全局变量或者静态变量 static  SingalTon * _singleTon=nil; +(SingalTon *) ShareSingleTon { //    返回对象前需要判断,这个对象之前是否创建过,如果没有创建过,就需要创建一个对象,如果创建过,就把上一次创建的对象返回出去 // 多个线程同时访问单例类时,就会创建多个单例类,就

java 核心学习笔记(四) 单例类

如果一个类始终只能创建一个实例,那么这个类被称作单例类. 一些特殊的应用场景可能会用到,为了保证只能创建一个实例,需要将构造方法用private修饰,不允许在类之外的其它地方创建类的实例. 又要保证可以创建该类的实例,所以需要一个public方法来创建该类的实例,由于在创建之前并没有该类的对象存在,所以只能使用类来调用此方法,故用static修饰. 同时需要将该类创建得到的实例分配一个引用才能知道该实例是否存在与是否唯一,所以需要一个成员变量保存创建的对象,同时由于该对象由上述静态方法创建,则该