单例模式
程序员文章站
2024-01-01 21:49:52
一、单例模式 单例模式分为四种:1.文件,2.类,3.基于__new__方法实现单例模式,4.基于metaclass方式实现 2.类实现如下: class Sigletion(objects): import time def __init__(self): time.sleep(1) @class ......
单例模式:顾名思义就是对象的实例只有一个
实现单例模式的方式有多种方式
1、饿汉式
// 饿汉式单例
public class Singleton1 {
// 私有构造
private Singleton1() {}
private static Singleton1 single = new Singleton1();
// 静态工厂方法
public static Singleton1 getInstance() {
return single;
}
}
饿汉式是在类加载初始化的时候就已经吧对象创建好了,提供了一个静态的工厂方法供使用者调用,除非系统重启不然对象不会消失,本身也是线程安全的。但是有一个缺点就是不管后面用不用这个对象都会被创建。
2、.懒汉式
// 懒汉式单例
public class Singleton2 {
// 私有构造
private Singleton2() {}
private static Singleton2 single = null;
public static Singleton2 getInstance() {
if(single == null){
single = new Singleton2();
}
return single;
}
}
实现了使用时才能创建单例对象,但是存在现在安全问题,多线程时会实例化多个对象,不能保证单例,一般都会用synchronizeds与valatile来保证其线程安全
public class Singleton3 {
// 私有构造
private Singleton3() {}
private volatile static Singleton3 single = null;
// 双重检查
public static Singleton3 getInstance() {
if (single == null) {
synchronized (Singleton3.class) {
if (single == null) {
single = new Singleton3();
}
}
}
return single;
}
}
关于valatile的作用有禁止指令重排序、内存可见性。百度一下二者作用再结合下面的话就能知道
多个线程同时对single访问时虽然有synchronized关键字但是当第一个线程创建对象后,jvm会给数据分配内存,在此过程中可能第二个线程已经进来,此时对于第二个线程来说single依旧是null,依旧会再去创建。valatile关键字会保证多线程场景下一个线程对数据改变,其他线程立刻可见。所以要加valatile关键字
除了上面二种常用的,下面也有几种可以实现
3、静态内部类
public class Singleton3 {
// 私有构造
private Singleton3() {}
// 静态内部类
private static class InnerObject{
private static Singleton3 single = new Singleton3();
}
public static Singleton3 getInstance() {
return InnerObject.single;
}
}
4、静态代码块
public class Singleton4 {
// 私有构造
private Singleton4() {}
private static Singleton4 single = null;
// 静态代码块
static{
single = new Singleton4();
}
public static Singleton4 getInstance() {
return single;
}
}
另外在spring中也有不少地方用到单例模式,spring的依赖注入,AOP的切点定义中等。
本文地址:https://blog.csdn.net/woxingyixu/article/details/111885295