美文网首页
11-并发下的单例

11-并发下的单例

作者: 加碘盐_ed6c | 来源:发表于2018-05-06 17:23 被阅读0次

单例

我们都知道面向对象程序设计中,最让人着迷和惊叹的就是设计模式了,在学习Java中最常用的就是单例模式了。那么什么是单例模式呢?所谓单例模式就是我们一个对象只能有一个实例。这么说不是很确切的话还可以这么说,我们经常在windows下可以看到一个应用只能打开一个,再次打开的时候不会产生新的进程。或者大家都喜欢玩游戏,很多游戏都不能双开的,这其中的原理就是单例模式。

用了这么多语言解决单例模式,那么怎么样实现单例模式呢?下面是一个简单的例子。

public class Main {
    private Main(){}
    static Main instance=new Main();
    static Main getInstance(){
        return instance;
    }
}

我们来分析一下这个单例模式,先建立了一个私有的默认构造方法,这就保证了在这个类的外部是无法new出这个类的实例的,之后我们在这个类中new出这个类的实例,用一个静态方法去返回这个例子。这样我们只有调用getInstance这个方法的时候,才能new出实例。那么这个单例是否有缺陷呢?当然是有的,我们使用了静态的变量区new,这就导致了在类加载的时候实例已经被new了,只是我们不能获得而已。那么我们接下来就改进一下这个单例模式。

public class Main {
    private Main(){}
    static class T{
        static Main instance=new Main();
    }
    static Main getInstance(){
        return T.instance;
    }
}

这个是改进好的代码,这个单例模式中,我们使用了一个静态内部类,但是不会再类加载的时候加载,只是在调用内部类的时候加载,也就是说,这个方式可以延迟类的加载,有的时候回带来一定的好处,不过就是比较麻烦,在对空间没有太大要求的时候,就可以不用这种方式。

并发下的单例

然而不管是第一种单例还是第二种单例,都是含有缺陷的,在单进程中,这个缺陷是不存在的,但是在高并发下就会出现。因为高并发的时候回存在两个线程同时调用一个方法,这个时候单例就不在单例了,会产生一个以上的实例。对于这个问题,单例模式就要改进一下了。

首先我们先看两个关键字:

  1. synchronized
  2. volatile

对于这两个关键字,看过前面内容的人应该很明白,是什么了吧?

  • synchronized:同步锁,用来解决线程之间同步的问题,他会给线程正在访问的资源加锁使线程更加安全。
  • volatile:这个可以说是一个轻量的锁的实现,但是本质不是锁,不过他有一个作用是使指令不重排序,也就是按照要求一步一步执行应该执行的指令。为什么要禁止指令重排序呢?在Java程序中我们new一个实例的时候,也许只是一句话,但是在CPU中执行的时候确实很多步骤,所以多线程并发先,就会造成new出对象的不完整性。

这两个关键字很关键哦
下面我们来写一个实例:

class A{
    private static volatile A instance;
    private static ReentrantLock lock=new ReentrantLock();
    private A(){
        System.out.println("Hello");
    }
    public static Singleton getInstance(){
        if(instance==null){
            synchronized (A.class){
                if(instance==null){
                    instance=new A();
                }
            }
        }
        return null;
    }
}

然后我们去使用多线程执行他:

public class Singleton  extends Thread{
    public void run(){
        super.run();
        System.out.println("MyThread");
        A.getInstance();
    }
    public static void main(String[] args){
        Singleton s1=new Singleton();
        Singleton s2=new Singleton();
        Singleton s3=new Singleton();
        s1.start();
        s2.start();
        s3.start();
    }
}

最后出现的结果就是在控制台只会输出一个hello,这就是唯一的一个实例。

相关文章

  • 11-并发下的单例

    单例 我们都知道面向对象程序设计中,最让人着迷和惊叹的就是设计模式了,在学习Java中最常用的就是单例模式了。那么...

  • 11-单例模式

  • Spring 单例 Bean 与多线程深度分析

    Spring 的 bean默认是单例的,在高并发下,如果在 Spring 的单例 bean 中设置成员变量,则会发...

  • 多线程并发下的单例模式

    定义: 单例模式是设计模式中最简单的形式之一。这一模式的目的是使得类的一个对象成为系统中的唯一实例。 下面通过代码...

  • 2019-07-31 你真的理解单例模式吗?

    高并发下线程安全的单例模式(最全最经典) https://blog.csdn.net/cselmu9/articl...

  • Android开发学习——Day10(单例设计模式&实战:扑克游

    学习目的 1.学习单例设计模式 2.完善并练习实战项目:扑克游戏 学习过程 了解单例设计模式,并简单运用。完善之前...

  • Java & Groovy & Scala & Kotlin -

    Scala 篇 单例对象 在 Java 中实现单例对象通常需要自己实现一个类并创建 getInstance() 的...

  • 好未来

    1.设计并实现单例模式 原理: ①单例类只能有一个实例 ②这个实例只能由单例类来生成 ③必须给所有其他对象提供这个...

  • 单例

    单例 单例是一种设计模式,确保一个类最多同时存在一个实例,并易于访问。 单例实现 1、私有构造方法 2、内部键静态...

  • 设计模式

    1.设计并实现单例模式 要求:①单例类只能有一个实例②这个实例只能由单例类来生成③必须给所有其他对象提供这个实例目...

网友评论

      本文标题:11-并发下的单例

      本文链接:https://www.haomeiwen.com/subject/rguurftx.html