今天看到群里,关于单例模式的多线程下的安全问题:
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提交多个结果的可能,有时候在去掉重复记录上很有用(如果记录的格式比较固定的话)。
相关推荐
单例模式单例模式单例模式单例模式单例模式单例模式单例模式单例模式
C#单例模式C#单例模式详解C#单例模式详解C#单例模式详解
单例模式详解~~单例模式详解~~单例模式详解~~
2020-02-10 王争设计模式之美进入课程讲述:冯永吉时长 10:21大小 8.31M上两节课中,我们针对单例模式,讲解了单例的应用场景、几种常见的代码实现
首先向关注过我这个系列...这立刻让我想到了最常用也是最简单最容易理解的一个设计模式 单例模式 何为 单例模式 ? 故名思议 即 让 类 永远都只能有一个实例。 由于 示例代码 比较简单 我也加了注释,这里就不在赘述
单例模式是最简单的设计模式之一,但是对于Java的开发者来说,它却有很多缺陷。在本月的专栏中,David Geary探讨了单例模式以及在面对多线程(multithreading)、类装载器(classloaders)和序列化(serialization)时...
单例模式的七种实现方法以及分析,可以作文大作业提交 1.前言 4 1.1 课题的研究背景 4 1.2 课题主要研究目标 4 2.相关技术简介 4 2.1Java简介 4 2.2IDEA简介 4 3. 单例模式的7种实现方式 5 3.1饿汉式(使用静态常量...
一个简单的java工程,包含注释,一目了然,其中包含了单例模式的所有实现方式,懒汉式,饿汉式,双重校验,枚举,静态内部类等方式实现单例。
单例模式--只能弹出一个窗体 只能弹出一个窗体
设计模式之七种单例模式代码及ppt,包含多线程环境测试和反序列化测试
单例模式和工厂模式结合应用,实现了产品的生产,适合用做课程设计,包含详细文档和代码。Java语言。喜欢的可以下载来看看那
几种单例模式的书写方式
使用单例模式创建学生管理系统(饿汉式、懒汉式)
php单例模式php单例模式php单例模式php单例模式
模式就像是OOP开发人员的配方,每种模式都提供了所需的成分。模式对OOP开发人员尤其有用,因为它有助于创建稳定的API,...本章将介绍两种常用的模式:单例模式和工厂模式。前者被称为类的职责,后者被称为类的多态性。
这个讲的是单例模式的多种不同实现方式,希望对单例感兴趣的同学看看
是http://blog.csdn.net/lxq_xsyu/article/category/1544127博客中java设计模式的源代码。下载前请先看《设计模式——单例模式》一文。
Java SE程序 单例模式Java SE程序 单例模式Java SE程序 单例模式Java SE程序 单例模式Java SE程序 单例模式Java SE程序 单例模式Java SE程序 单例模式Java SE程序 单例模式Java SE程序 单例模式Java SE程序 单例模式...
设计模式C++学习之单例模式(Singleton)
此示例展示了Qml 的单例模式(类似全局对象,只生成一次实例,可全局使用) surfsky.cnblogs.com