是的,Spring做些 魔术 。查看Spring
Docs:
这就是神奇的地方:所有
@Configuration类在启动时都使用CGLIB进行了子类化。在子类中,子方法在调用父方法并创建新实例之前,首先检查容器中是否有任何缓存(作用域)的bean。
这意味着
@Bean通过CGLIB代理对方法的调用,因此将返回Bean的缓存版本(不创建新的)。
@Beans 的默认范围是
SINGLETON,如果您指定其他范围,例如
PROTOTYPE调用将传递给原始方法。
请注意,这 对于静态方法无效 。根据春季文件:
由于技术限制,对静态
@Bean方法的调用永远不会被容器拦截,甚至在@Configuration类内也不会(如本节前面所述),因为技术限制:CGLIB子类只能覆盖非静态方法。因此,直接调用另一个@Bean方法具有标准的Java语义,从而导致直接从工厂方法本身直接返回一个独立的实例。



