java多线程编程核心技术——第六章总结

1.0立即加载/“饿汉式”

  立即加载:实用类的时候已经将对象创建完毕,常见的实现方法就是直接new实例化。

  注:是在调用方法前,就已经实例化了(通常是在类一加载的时候就已经实例化了)。

public class Singleton {
    //立即加载方法==饿汉式
    private static Singleton singleton = new Singleton();

    private Singleton() {
    }

    public static Singleton getInstance() {
        //此代码版本为立即加载
        //此代码版本缺点:是不能有其他实例变量
        //因为getInstance()方法没有同步
        //所以可能出现非线程安全问题
        return singleton;
    }
}

2.0延迟加载/“懒汉式”

  延迟加载:在调用get()方法时实例才被创建,常见的实现办法就是在get()中进行实例化。

  注:在调用获取实例的方法时,实例才被创建。

public class Singleton {
    private static Singleton singleton;

    private Singleton(){}

    public static Singleton getInstance() {
        //延迟加载,若多个线程在此if失去执行权,最终会产生很多实例对象
        if (singleton == null) {
            singleton = new Singleton();
        }
        return singleton;
    }
}

  可以使用synchronized方法或者synchronized同步代码块解决多线程中的同步问题,但是效率较低。

  书中提供了双检查锁机制:

  可以使用DCL双检查锁机制实现单例:

public class Singleton {
    private volatile static Singleton singleton;

    private Singleton(){}
    //使用双检查锁机制来解决问题,既保证了不需要使用同步代码块的异步执行性,又保证了单例效果。

    public static Singleton getInstance() {
        if (singleton == null) {
            synchronized (Singleton.class) {
                if (singleton == null) {
                    singleton = new Singleton();
                }
            }
        }
        return singleton;
    }
}

  注:关于双检索的争议还是有很多的,但是目前可以认为其实安全的。

  双检查锁的安全性问题总结(未完成)

3.0使用静态内置类实现单例模式

public class Singleton {
    //内部类形式
    private static class MyHanlder {
        private static Singleton singleton = new Singleton();
    }

    private Singleton() {}

    public static Singleton getInstance() {
        return MyHanlder.singleton;
    }
}

4.0序列化与反序列化的单例模式实现

  虽然静态内置类可以达到线程安全,但如果遇到序列化对象,使用默认的方式运行得到的还是多例。

public class Singleton implements Serializable{
    private static final long seroalVersionUID = 888L;

    //内部类形式
    private static class MyHanlder {
        private static Singleton singleton = new Singleton();
    }

    private Singleton() {}

    public static Singleton getInstance() {
        return MyHanlder.singleton;
    }

    protected Object readResolve() throws ObjectStreamException {
        return MyHanlder.singleton;
    }
}

  书写readResolve()方法就可以避免在序列化与反序列化的过程中单例失效。

  注:目前为什么这样还不理解。

5.0使用static代码块实现单例模式

  静态代码块中的代码在使用类的时候就已经执行了,所以可以应用静态代码块的这个特性来实现单例设计模式。

public class Singleton implements Serializable{
    private static Singleton singleton = null;

    private Singleton() {}

    static {
        singleton = new Singleton();
    }

    public static Singleton getInstance() {
        return singleton;
    }
}

6.0使用enum枚举数据类型实现单例模式

  枚举与静态代码块类似,在使用枚举时,构造方法会被自动调用,也可以使用其这个特性实现单例设计模式。

本文内容是书中内容兼具自己的个人看法所成。可能在个人看法上会有诸多问题(毕竟知识量有限,导致认知也有限),如果读者觉得有问题请大胆提出,我们可以相互交流、相互学习,欢迎你们的到来,心成意足,等待您的评价。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏海说

Java源码学习 -- java.lang.StringBuilder,java.lang.StringBuffer,java.lang.AbstractStringBuilder

一直以来,都是看到网上说“ StringBuilder是线程不安全的,但运行效率高;StringBuffer 是线程安全的,但运行效率低”,然后默默记住:一个是...

1630
来自专栏猿说1024

Java 单例模式

915
来自专栏向治洪

单例模式

单例对象(Singleton)是一种常用的设计模式。在Java应用中,单例对象能保证在一个JVM中,该对象只有一个实例存在。 这样的模式有几个好处: 1、某些...

1727
来自专栏Java后端技术

你敢说自己了解单例模式?

  最近在学习设计模式,在看到单例模式的时候,我一开始以为直接很了解单例模式了,实现起来也很简单,但是实际上单例模式有着好几个变种,并且多线程中涉及到线程安全问...

782
来自专栏用户画像

JAVA单例模式

在整个应用中,保证一个类只有一个实例,它提供了一个可以访问到它自己的全局访问点(静态方法)。

672
来自专栏高爽的专栏

单例模式

概述 单例模式(Singleton Pattern),创建对象模式之一,单例模式确保一个类只能创建一个实例。单例模式适用于一个系统要求一个类只有一个实例时,比如...

1890
来自专栏calmound

设计模式--单例模式Singleton

单例模式顾名思义整个程序下只有一个实例,例如一个国家只有一个皇帝,一个军队只有一个将军。 单例模式的书写又分为饿汉模式和懒汉模式 饿汉模式    类中代码 pa...

2719
来自专栏风口上的猪的文章

.NET面试题系列[8] - 泛型

“可变性是以一种类型安全的方式,将一个对象作为另一个对象来使用。“ - Jon Skeet

1063
来自专栏Nian糕的私人厨房

ECMAScript6 解构赋值

在 ES6 中,关于解构的含义为:允许按照一定模式,从数组和对象中提取值,对变量进行赋值,而数组、对象和字符串,都能通过这种方式进行赋值

634
来自专栏JAVA高级架构

面试中单例模式有几种写法

“你知道茴香豆的‘茴’字有几种写法吗?” 纠结单例模式有几种写法有用吗?有点用,面试中经常选择其中一种或几种写法作为话头,考查设计模式和coding style...

2606

扫码关注云+社区