单例模式---JAVA

简介: “饿汉”模式“懒汉”模式

单例模式:保证某个类在程序中只存在唯一一份实例, 而不会创建出多个实例。

单例模式可以通过实例创建的时间来分为两种:“饿汉”和“懒汉”模式。

“饿汉”模式

所谓的“饿汉”模式实则就是在类加载的时候创建出实例。

首先我们先创建一个类Singleton再在类中写一个静态私有的常量,而这个常量的值就是唯一对象的引用。

classSingleton{
privatestaticfinalSingletonsingleton=newSingleton();
}

image.gif

因为这个唯一对象是私有的所以还需要一个get方法。

publicstaticSingletongetSingleton() {
returnsingleton;
}

image.gif

可是现在我们从别的地方还是可以直接new出这个类的其它实例,这个该怎么解决呢?我们只需要再写一个私有的构造方法就可以解决了。

privateSingleton() {}

image.gif

image.png

这就是一个简单的单例模式了(“饿汉”模式)

完整代码

classSingleton{
privatestaticfinalSingletonsingleton=newSingleton();
publicstaticSingletongetSingleton() {
returnsingleton;
    }
privateSingleton() {}
}

image.gif

但是由于“饿汉”模式的实例是在类加载时就创建了,并没有考虑这个实例在代码中是否使用,这就有可能会导致代码中并没有用这个类可是你却已经创建了,这就会导致内存浪费,解决办法就是“懒汉”模式。

“懒汉”模式

“懒汉”模式是在线程的一次调用该类的get方法时进行唯一实例的创建。

先创建一个类,该类中有一个私有的类属性,该属性的值为null或唯一实例的引用。

classSingleton{
privatestaticSingletonsingleton=null;
}

image.gif

为了保证实例的唯一性,将构造方法写为私有的。

privateSingleton() {}

image.gif

写一个get方法,该方法在第一次被调用时会创建出一个唯一实例。

publicstaticSingletongetSingleton() {
if (singleton==null) {
singleton=newSingleton();
    }
returnsingleton;
}

image.gif

这个get方法在单线程中看是没有任何问题的,但是如果放在多线程代码中就会出现线程安全问题,例如如果出现以下的执行顺序那么就不是单例模式了。

image.png

解决办法就是加锁

publicstaticSingletongetSingleton() {
synchronized (Singleton.class) {
if (singleton==null) {
singleton=newSingleton();
        }
    }
returnsingleton;
}

image.gif

但是此时代码又面临了一个效率问题,由于我们只有第一次调用get时才会创建实例才会出现线程安全问题,可是现在我们每次调用get方法都会进行加锁操作,而加锁就会有锁竞争从而导致代码效率过低的问题,解决方法就是再加一层 if 判断。


publicstaticSingletongetSingleton() {
if (singleton==null) {
synchronized (Singleton.class) {
if (singleton==null) {
singleton=newSingleton();
            }
        }
    }
returnsingleton;
}

image.gif

因为new操作很有可能触发指令重排序,所以为了防止编译器对其进行优化建议加上volatile

privatestaticvolatileSingletonsingleton=null;

image.gif

完整代码

classSingleton{
privatestaticvolatileSingletonsingleton=null;
privateSingleton() {}
publicstaticSingletongetSingleton() {
if (singleton==null) {
synchronized (Singleton.class) {
if (singleton==null) {
singleton=newSingleton();
                }
            }
        }
returnsingleton;
    }
}

image.gif


目录
相关文章
|
7月前
|
设计模式 安全 Java
JAVA设计模式1:单例模式,确保每个类只能有一个实例
JAVA设计模式1:单例模式,确保每个类只能有一个实例
|
3天前
|
设计模式 安全 Java
【JAVA】Java 中什么叫单例设计模式?请用 Java 写出线程安全的单例模式
【JAVA】Java 中什么叫单例设计模式?请用 Java 写出线程安全的单例模式
|
29天前
|
SQL 设计模式 安全
Java单例模式几种写法以及代码案例拿来直接使用
Java单例模式几种写法以及代码案例拿来直接使用
32 0
|
16天前
|
设计模式 存储 Java
Java设计模式:解释一下单例模式(Singleton Pattern)。
`Singleton Pattern`是Java中的创建型设计模式,确保类只有一个实例并提供全局访问点。它通过私有化构造函数,用静态方法返回唯一的实例。类内静态变量存储此实例,对外仅通过静态方法访问。
16 1
|
5月前
|
设计模式 Java
Java设计模式【一】:单例模式
Java设计模式【一】:单例模式
26 0
|
2月前
|
设计模式 安全 Java
Java设计模式之单例模式
在软件工程中,单例模式是一种常用的设计模式,其核心目标是确保一个类只有一个实例,并提供一个全局访问点来获取这个实例。Java作为一门广泛使用的编程语言,实现单例模式是面试和实际开发中的常见需求。
66 9
Java设计模式之单例模式
|
3月前
|
设计模式 Java 安全
[Java]单例模式
本篇文章主要阐述单例模式的基础实现,重心在于如何解决单例模式中的“线程安全”问题。不涉及理论。 如果文中阐述不全或不对的,多多交流。
80 0
[Java]单例模式
|
3月前
|
Java
JAVA基础--内部类和静态内部类、单例模式---静态内部类剖析
JAVA--内部类和静态内部类、单例模式---静态内部类剖析
|
3月前
|
存储 缓存 Java
Java volatile关键字-单例模式的双重锁为什么要加volatile
Java volatile关键字--单例模式的双重锁为什么要加volatile
51 10
|
4月前
|
设计模式 安全 Java
Java设计模式—单例模式的实现方式和使用场景
那么为什么要有单例模式呢?这是因为有的对象的创建和销毁开销比较大,比如数据库的连接对象。所以我们就可以使用单例模式来对这些对象进行复用,从而避免频繁创建对象而造成大量的资源开销。
60 1