单例模式的特点

  • 一个类只允许产生一个实例化对象。
  • 单例类构造方法私有化,不允许外部创建对象。
  • 单例类向外提供静态方法,调用方法返回内部创建的实例化对象。

懒汉式(线程不安全)

其主要表现在单例类在外部需要创建实例化对象时再进行实例化,进而达到Lazy Loading 的效果。

通过静态方法 getSingleton() 和private 权限构造方法为创建一个实例化对象提供唯一的途径。

不足:未考虑到多线程的情况下可能会存在多个访问者同时访问,发生构造出多个对象的问题,所以在多线程下不可用这种方法。

/**
* @author MrRoot
* @since 2018-12-17
* 懒汉式(线程不安全)
*/
public class Singleton {
private static Singleton singleton; private Singleton(){ } public static Singleton singleton(){
if (singleton == null){
singleton = new Singleton();
}
return singleton;
}
}

懒汉式(线程安全,同步方法,不推荐使用)

针对懒汉式的线程不安全,自然会想到给 getSingleton() 进行 synchronized 加锁来保证线程同步。

不足:效率低。大多数情况下这个锁占用的额外资源都浪费了,每个线程在想获得类的实例时候,执行 getSingleton() 方法都要进行同步。

/**
* @author MrRoot
* @since 2019-3-27
* 懒汉式(线程安全,同步方法,不推荐使用)
*/
public class Singleton {
private static Singleton singleton; private Singleton(){ } public static synchronized Singleton singleton(){
if (singleton == null){
singleton = new Singleton();
}
return singleton;
} }

饿汉式(线程安全)

在进行类加载时完成实例化对象的过程就是饿汉式的形式。

避免了线程同步问题,在运行这个类的时候进行加载,之后直接访问

不足:相比接下来的静态内部类而言,这种方法比静态内部类多了内存常驻,容易造成内存浪费,也未达到延迟加载的效果。

/**
* @author MrRoot
* @since 2019-3-27
* 饿汉式(线程安全)
*/
public class Singleton{
private static Singleton singleton = new Singleton(); private Singleton(){ } public static Singleton singleton(){
return singleton;
}
}

静态内部类加载(线程安全)

静态内部类不会在单例加载时加载,当调用 getSingleton() 方法时才会进行加载,达到类似懒汉式效果,并且也是线程安全的。

类的静态属性只会在第一次加载类时进行初始化,所以上面的方法JVM 帮助我们保证了线程的安全性,在类进行初始化时,其他线程无法进入。

/**
* @author MrRoot
* @since 2019-3-27
* 静态内部类加载(线程安全)
*/
public class Singleton{
private static Singleton singleton; private static class SingletonInner{
private static final Singleton instance = new Singleton();
} public static Singleton getSingleton(){
return SingletonInner.instance;
}
}

枚举(线程安全)

自由串行化;保证只有一个实例;线程安全。

Effective Java 作者所提倡的方法,近乎完美,在继承场景下不适用。

/**
* @author MrRoot
* @since 2019-3-27
* 枚举(线程安全)
*/
enum Singleton{
INSTANCE; public void method(){ }
} class Test{
public static void main(String[] args) {
Singleton.INSTANCE.method();
}
}

懒汉式双重校验锁法(通常线程安全,不可保证完全安全)

使用同步代码块避免了第二种方法的效率低的问题,但此方法并不能完全起到线程同步的作用,与上面第一种方法产生的问题相似,多线程访问时可能产生多个对象。

/**
* @author MrRoot
* @since 2019-3-27
* 懒汉式双重校验锁法(通常线程安全,不可保证完全安全)
*/
class Singleton{
private static Singleton singleton; private Singleton(){ } public static Singleton singleton(){
if (singleton == null){
synchronized (Singleton.class){
if (singleton == null){
singleton = new Singleton();
}
}
}
return singleton;
}
}

懒汉式双重检查终极版

与第六种方法不同的是,此方法给singleton 的声明上加了关键字 volatile ,进而解决了低概率的线程不安全问题。

volatile 起到禁止指令重排的作用,在它赋值完成之前,就不会调用读操作(singleton == null)。

/**
* @author MrRoot
* @since 2019-3-27
* 懒汉式双重检查终极版(volatile)
*/
class Singleton{
private static volatile Singleton singleton; private Singleton(){ } public static Singleton singleton(){
if (singleton == null){
synchronized (Singleton.class){
if (singleton == null){
singleton = new Singleton();
}
}
}
return singleton;
}
}

使用 ThreadLocal 实现(线程安全)

ThreadLocal 会为每一个线程提供一个独立的变量副本,从而隔离了多个线程对数据的访问冲突。

对于多线程资源共享的问题,同步机制采用了“以时间换空间”的方式,而ThreadLocal 采用了“以空间换时间”的方式。前者仅提供一份变量,让不同的线程排队访问,而后者为每一个线程都提供了一份变量,因此可以同时访问而互不影响。

/**
* @author MrRoot
* @since 2019-3-27
* 使用 ThreadLocal 实现(线程安全)
*/
class Singleton{
private static final ThreadLocal<Singleton> singleton = new
ThreadLocal<Singleton>(){
@Override
protected Singleton initialValue(){
return new Singleton();
}
}; private Singleton(){ } public static Singleton getSingleton(){
return singleton.get();
}
}

使用CAS 锁实现(线程安全)

/**
* @author MrRoot
* @since 2019-3-27
* 使用 CAS 实现(线程安全)
*/
public class Singleton {
private static final AtomicReference<Singleton> INSTANCE = new AtomicReference<Singleton>(); private Singleton(){ } public static final Singleton getSingleton(){
for (;;){
Singleton current = INSTANCE.get();
if (current != null){
return current;
}
current = new Singleton();
if (INSTANCE.compareAndSet(null,current)){
return current;
}
}
} public static void main(String[] args) {
Singleton singleton1 = Singleton.getSingleton();
Singleton singleton2 = Singleton.getSingleton();
System.out.println(singleton1 == singleton2);
}
}

9种Java单例模式详解的更多相关文章

  1. 9种Java单例模式详解(推荐)

    单例模式的特点 一个类只允许产生一个实例化对象. 单例类构造方法私有化,不允许外部创建对象. 单例类向外提供静态方法,调用方法返回内部创建的实例化对象.  懒汉式(线程不安全) 其主要表现在单例类在外 ...

  2. Java 单例模式详解

    概念: java中单例模式是一种常见的设计模式,单例模式分三种:懒汉式单例.饿汉式单例.登记式单例三种. 单例模式有一下特点: 1.单例类只能有一个实例. 2.单例类必须自己自己创建自己的唯一实例. ...

  3. 【JAVA单例模式详解】

    设计模式是一种思想,适合于任何一门面向对象的语言.共有23种设计模式. 单例设计模式所解决的问题就是:保证类的对象在内存中唯一. 举例: A.B类都想要操作配置文件信息Config.java,所以在方 ...

  4. Java 单例模式详解(转)

    概念: java中单例模式是一种常见的设计模式,单例模式分三种:懒汉式单例.饿汉式单例.登记式单例三种. 单例模式有一下特点: 1.单例类只能有一个实例. 2.单例类必须自己自己创建自己的唯一实例. ...

  5. java单例模式详解

    饿汉法 饿汉法就是在第一次引用该类的时候就创建对象实例,而不管实际是否需要创建.代码如下: public class Singleton { private static Singleton = ne ...

  6. android java 设计模式详解 Demo

    android java 设计模式详解 最近看了一篇设计模式的文章,深得体会,在此基础我将每种设计模式的案例都写成Demo的形式,方便读者研究学习, 首先先将文章分享给大家: 设计模式(Design ...

  7. Java内部类详解

    Java内部类详解 说起内部类这个词,想必很多人都不陌生,但是又会觉得不熟悉.原因是平时编写代码时可能用到的场景不多,用得最多的是在有事件监听的情况下,并且即使用到也很少去总结内部类的用法.今天我们就 ...

  8. Java虚拟机详解----JVM常见问题总结

    [声明] 欢迎转载,但请保留文章原始出处→_→ 生命壹号:http://www.cnblogs.com/smyhvae/ 文章来源:http://www.cnblogs.com/smyhvae/p/4 ...

  9. [转] Java内部类详解

    作者:海子 出处:http://www.cnblogs.com/dolphin0520/ 本博客中未标明转载的文章归作者海子和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置 ...

随机推荐

  1. apache-jmeter-3.1的简单压力测试使用方法

    压力测试工具LoadRunner是收费的,而且操作复杂.作为开发人员当然是用apache提供的jmeter,免费容易上手. jmeter下载地址http://jmeter.apache.org/首先下 ...

  2. web安全-点击劫持

    web安全-点击劫持 opacity=0 iframe是目标网站 被内嵌了 1.用户亲手操作 盗取用户 视频 2.用户不知情 >* 引导点击 其实点击的是覆盖在下面opacity=0的ifram ...

  3. linux文件系统相关资料

             linux下文件系统通常是通过虚拟文件系统(VFS)蔽下层具体文件系统操作的差异,为上层的操作提供一个统一的接口.文件系统底层都是用系统IO缓存层提供的块读写接口,实现逻辑块到物理块 ...

  4. 阅读GFS的一点总结

    这是我第一次阅读学术论文,文章中充斥的一些学术名词给我的阅读带来了一些困难,因为此前没有接触过这方面的内容,在同学的帮助下,查阅了一些资料,终于对GFS有了一点认识,写出这一些感悟,文章措辞不严谨之处 ...

  5. P2245 星际导航 瓶颈路

    \(\color{#0066ff}{ 题目描述 }\) sideman 做好了回到 \(\text{Gliese}\) 星球的硬件准备,但是 \(\text{sideman}\) 的导航系统还没有完全 ...

  6. Tensorflow方法介绍

    一.reduce系列函数(维度操作) 1.tf.reduce_sum( input_tensor, axis=None, keep_dims=False, name=None, reduction_i ...

  7. opencv-图片合成视频

    无论视频的合成还是分解我们都需要进行解码器或者是编码器(因为视频不是一帧一帧进行存储的,而是进行过压缩编码.) import cv2 img = cv2.imread('image1.jpg') im ...

  8. 在服务器上使用 gradle 打包 android 源码

    安装 android-tools mkdir ~/android && cd ~/android   wget https://dl.google.com/android/reposi ...

  9. BZOJ1725】[Usaco2006 Nov]Corn Fields牧场的安排 状压DP

    Description Farmer John新买了一块长方形的牧场,这块牧场被划分成M列N行(1<=M<=12; 1<=N<=12),每一格都是一块正方形的土地.FJ打算在牧 ...

  10. linux虚拟机管理

    1.虚拟机管理命令virsh-manager        ##开启虚拟机管理器 virsh list          ##显示正在运行的虚拟机virsh list  --all     ##查看所 ...