单例模式的坑

先看一段代码

/**
 * @author xiaofei.wxf
 */
public class SingletonA {
    private SingletonA(){}

    static final SingletonB b = SingletonB.b;
    static final SingletonA a = new SingletonA();
}
/**
 * @author xiaofei.wxf
 */
public class SingletonB {
    private SingletonB(){}

    static final SingletonA a = SingletonA.a;
    static final SingletonB b = new SingletonB();
}

/**
 * @author xiaofei.wxf
 */
public class Singleton {
    public static void main(String[] args) {
        SingletonA a = SingletonA.a;
        SingletonB b = SingletonB.b;

        SingletonA a1 = SingletonB.b.a;
        SingletonB b1 = SingletonA.a.b;

        System.out.println(a == a1);
        System.out.println(b == b1);
    }
}

输出是什么?

false
true

你的答案也是这个吗,或许你是对的,但也许你是蒙的。好,让我们来分析下其中的坑。


平时我们写单例的时候,一般不会注意细节,以为只要是static字段,jvm总会帮我们安排好一切。但是这里不然,Why?


对于static字段,jvm加载类之后会调用类的<clinit>方法,这个方法我们是看不到的,是java编译器为我们添加的。它的作用正如名字所表示的那样,class init,用来初始化所加载的类。clinit方法会按照我们定义的顺序,一个个初始化static的代码块或者字段。


那让我们看下上面的代码执行的流程:

1. 加载SingletonA类

2. 执行SingletonA 的 static final SingletonB b = SingletonB.b;

3. 加载SingletonB类

4. 执行SingletonB 的 static final SingletonA a = SingletonA.a;

到这一步SingletonB.a已经出来了,为null

5. 执行SingletonB 的 static final SingletonB b = new SingletonB();

这一步SingletonA.b 和 SingletonB.b  为 第一步创建的对象

6. 执行SingletonA 的 static final SingletonA a = new SingletonA();

这一步SingletonA.a 为第六步创建的对象


最后,SingletonA.a != SingletonB.a (null)


知道了原理,我们来把程序改正确

/**
 * @author xiaofei.wxf
 */
public class SingletonA {
    private SingletonA(){}
    static final SingletonA a = new SingletonA();

    static final SingletonB b = SingletonB.b;
}
/**
 * @author xiaofei.wxf
 */
public class SingletonB {
    private SingletonB(){}
    static final SingletonB b = new SingletonB();

    static final SingletonA a = SingletonA.a;
}


好了,调下次序就好了。这个简单的坑告诉我们,static的顺序很重要,我们写代码的时候要避免循环引用的问题,最好把单例的初始化放到第一行static中去。


你以为这样就结束了?继续看


既然是单例了,我们对其他类的引用就没必要是static的,对吧?


好的,我们去掉其他字段的static修饰符

/**
 * @author xiaofei.wxf
 */
public class SingletonA {
    private SingletonA(){}
    static final SingletonA a = new SingletonA();

    final SingletonB b = SingletonB.b;
}
/**
 * @author xiaofei.wxf
 */
public class SingletonB {
    private SingletonB(){}
    static final SingletonB b = new SingletonB();

    final SingletonA a = SingletonA.a;
}


还是调用之前的main方法,输出结果是什么?

false
true
没错,又出错了。

按照上面的方法我们再分析下

1. 加载A类

2. 执行A的clinit,执行A的static块,这时需要创建A对象

3. 调用A的构造器init方法,此时需要用到类B

4. 加载B类

5. 执行B的clinit,执行B的static块,这事创建B的对象

6. 执行B的构造器init方法,给B.a赋值为这时的A.a(null)

7. 回到A的构造函数init方法将B.b复制给A.b

8. 将A实例赋值给A.a 

所以a1值为null


这个程序是一个典型的循环应用初始化顺序问题,这也就是为什么Spring的IOC需要有个init方法的原因(个人认为)。

本例的场景如果不改变设计思路是无法解决这个问题的。解决方法两种,1. 学Spring给类价格init,但是Spring是容器调用init,但这里,谁也不想使用前调用下init 2. 静态块

必须要改变设计思路,使用静态块来解决,方法如下:

/**
 * @author xiaofei.wxf
 */
public class SingletonA {
    private SingletonA(){}
    static final SingletonA a = new SingletonA();
    static {
        a.b  = SingletonB.b;
    }
    SingletonB b;
}
/**
 * @author xiaofei.wxf
 */
public class SingletonB {
    private SingletonB(){}
    static final SingletonB b = new SingletonB();
    static{
        b.a = SingletonA.a;
    }
    SingletonA a;
}


结论:不起眼的地方总会导致我们程序运行出问题。并不是我们不知道这个知识点,拆解分析下来我们也是知道原因的,就是当程序太复杂的时候很难定位的具体的问题点。

所以,写程序给自己多几条原则,根据原则写就不太会掉进已知的坑里面。






















<think>嗯,用户现在在问关于使用装饰器实现单例模式的问题,特别是提到装饰器里的程序先执行,之后是类中的,如果装饰器中先实例化一个对象,那么由于Python的__new__不会创建新对象。我需要仔细分析这个问题,确保回答准确且易懂。 首先,用户可能已经了解装饰器和单例模式的基本概念,但可能对它们的交互机制,尤其是Python内部如何通过__new__和装饰器来管理实例化过程不太清楚。用户提到装饰器里的程序先执行,这可能指的是装饰器在类定义时立即执行,而类实例化时则是在之后调用的时候。 我需要确认装饰器在单例模式中的工作方式。通常,使用装饰器实现单例时,装饰器会在类被定义时(也就是装饰器应用的时候)就创建一个实例,并在后续的实例化请求中返回这个已存在的实例。这时候,Python的__new__方法可能不会被再次调用,因为装饰器已经控制了实例的创建过程。 用户提到“如果装饰器中先实例化一个对象,那么由于Python内部机制__new__不会创建新对象”。这里可能存在一个误解。实际上,当装饰器管理单例时,每次调用类的时候,装饰器会检查是否已经有实例存在。如果有,就直接返回现有的实例;如果没有,才调用类的__new__和__init__方法创建新实例。因此,装饰器并不会完全绕过__new__方法,而是在第一次实例化时正常调用,后续的调用则直接返回已有的实例,避免重复创建。 接下来,我需要解释装饰器如何拦截实例化过程。当使用装饰器包装类时,装饰器通常会替换原来的类或者类的构造函数。例如,装饰器可能将类替换为一个工厂函数,该工厂函数负责检查实例是否存在,并控制实例的创建。这样,当用户尝试创建类的实例时,实际上是调用这个工厂函数,而工厂函数决定是否调用原始类的__new__方法。 另外,用户可能混淆了装饰器的执行时机。装饰器在类定义时执行,而不是在实例化时执行。因此,装饰器内部的代码会在导入或定义类的时候立即运行,准备好单例实例的管理逻辑。而类的__new__和__init__方法则是在实际创建实例的时候被调用,但由于装饰器的控制,只有在第一次创建实例时才会真正执行这些方法。 用户的问题可能源于对装饰器应用时机和实例化流程的理解不足。需要明确装饰器在类定义阶段的作用,以及它如何包装类,使得后续的实例化请求被装饰器拦截,从而管理单例实例。同时,需要澄清__new__方法在单例模式中的角色,即装饰器并不是绕过__new__,而是通过控制实例的创建次数来确保单例。 另外,可能需要举例说明装饰器实现单例的具体代码,展示装饰器如何存储实例,并在后续调用时返回已存在的实例。例如,一个常见的单例装饰器可能使用闭包或类属性来保存实例,并在每次实例化时检查该实例是否存在。 总结来说,用户的问题核心在于理解装饰器如何与Python的实例化机制(特别是__new__方法)交互来实现单例模式。需要解释装饰器在类定义时的执行,以及它如何管理实例的创建,确保只有一个实例存在,而后续的实例化请求直接返回该实例,避免再次调用__new__和__init__。</think>$\boxed{通俗版解析}$ **装饰器实现单例的本质是「提前抢跑」**: 当用装饰器包装类时,装饰器的代码会在**类定义阶段立刻执行**,而类的 `__new__` 和 `__init__` 是在**实例化时**才触发。这个时间差让装饰器能预先控制实例的生成。 --- $\boxed{举个🌰}$ **代码示例**: ```python def 单例装饰器(cls): print("【装饰器立刻执行】开始监控", cls.__name__) 实例仓库 = None # 先占个 def 包装后的类(*args, **kwargs): nonlocal 实例仓库 if 实例仓库 is None: # 首次创建 print("🚀调用原类的__new__方法创建实例") 实例仓库 = cls(*args, **kwargs) # 这里才会触发类的__new__ else: print("🎯已有实例,直接返回") return 实例仓库 return 包装后的类 @单例装饰器 # 当Python解析到这行时,装饰器立刻执行! class 总统: def __new__(cls, *args, **kwargs): print("👑总统类的__new__被调用") return super().__new__(cls) def __init__(self): print("📝总统就职典礼") # 此时装饰器已经执行完毕,总统类被替换为「包装后的类」 ``` **运行结果**: ```python print("第一次创建总统:") a = 总统() print("\n第二次创建总统:") b = 总统() ``` **输出**: ``` 【装饰器立刻执行】开始监控 总统 # 类定义时就打印! 第一次创建总统: 🚀调用原类的__new__方法创建实例 👑总统类的__new__被调用 📝总统就职典礼 第二次创建总统: 🎯已有实例,直接返回 ``` --- $\boxed{关键机制拆解}$ 1. **装饰器抢先执行**: - 当代码读到 `@单例装饰器` 时,**立刻**执行装饰器函数,把原类替换成包装函数 `包装后的类` - 此时 `实例仓库` 变量被初始化为 `None`(占) 2. **首次实例化触发__new__**: - 当你第一次写 `总统()` 时,实际调用的是 `包装后的类()` - 检查到 `实例仓库` 为空,**才真正调用原类的 `__new__` 和 `__init__`** 3. **后续调用直接拦截**: - 第二次调用 `总统()` 时,`实例仓库` 已有值,直接跳过 `__new__` 和 `__init__` --- $\boxed{你的理解正确!}$ **你的观察非常准确**: - 装饰器在类定义阶段就「架设好了拦截网」 - 如果装饰器**提前创建实例**(比如把 `实例仓库 = cls()` 放在装饰器内部),那么连第一次调用 `总统()` 都不会触发 `__new__`,因为实例在装饰阶段就生成好了 --- $\boxed{对比实验}$ **修改装饰器提前创建实例**: ```python def 单例装饰器(cls): 实例仓库 = cls() # 立刻创建实例!此时会触发__new__和__init__ def 包装后的类(): return 实例仓库 return 包装后的类 @单例装饰器 class 总统: def __new__(cls): print("👑总统类的__new__被调用") return super().__new__(cls) def __init__(self): print("📝总统就职典礼") # 输出: 👑总统类的__new__被调用 # 类定义时就打印! 📝总统就职典礼 总统() # 这次调用不会触发任何__new__/__init__ ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值