- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
免责声明:我读过以下关于 JDK 动态代理和 CGLIB 的有用文章:https://stackoverflow.com/a/21762454/2674303
我读过以下有趣的文章:Injecting Spring Prototype bean into Singleton bean
原型(prototype):
@Service
@Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE)
class MessageBuilder {
private static final AtomicInteger instanceCounter = new AtomicInteger(0);
MessageBuilder() {
instanceCounter.incrementAndGet();
}
static int getInstanceCounter() {
return instanceCounter.get();
}
....
}
单例:
@Service
class MessageService {
private final MessageBuilder messageBuilder;
MessageService(MessageBuilder messageBuilder) {
this.messageBuilder = messageBuilder;
}
Message createMessage(String content, String receiver) {
return messageBuilder
.withContent(content)
.withReceiver(receiver)
.build();
}
}
测试:
@RunWith(SpringRunner.class)
@SpringBootTest
public class MessageServiceTest {
@Autowired
private MessageService messageService;
@Test
public void shouldCreateTwoBuilders() throws Exception {
//when
messageService.createMessage("text", "alice");
messageService.createMessage("msg", "bob");
//then
int prototypeCounter = MessageBuilder.getInstanceCounter();
assertEquals("Wrong number of instances", 2, prototypeCounter);
}
}
显然测试失败了,因为注入(inject)只发生一次,实际结果为 1,但我们预期为 2。
第二种情况:
Singleton 和 Test 是相同的,但原型(prototype)现在看起来像这样(proxyMode 已更改):
@Service
@Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE,
proxyMode = ScopedProxyMode.TARGET_CLASS)
class MessageBuilder {
// ...
}
当我们开始测试时,我们看到实际结果是 6,因为内部 createMessage
方法 messageBuilder 被访问了 3 次。并且 createMessage
方法被调用两次,因此 3*2=6。
为了解释行为,作者提供了下图:
我无法理解哪个 bean 是依赖的,以及为什么每次访问代理 messageBuilder
基因都会产生新的 bean 实例化。为什么第一种情况的情况不同?你能解释一下吗?据我了解 - 代理无论如何都是装箱的 - 使用 CGLIB 或使用动态代理,所以无论如何代理都会被注入(inject)
最佳答案
如果您使用 prototype
范围定义一个 bean,则当从 Spring 上下文引用该 bean 时,ApplicationContext
将返回一个新实例。在第一个示例中,当创建 MessageService
singleton bean 时,会创建 MessageBuilder
prototype 的新实例。然而,由于 MessageService
在 Spring 生命周期中仅构造一次(因为它是单例),因此它只请求注入(inject) MessageBuilder
原型(prototype) bean 的一个引用。
换句话说,在第一个示例中,MessageBuilder
bean 仅实例化一次,因为它被注入(inject)( Autowiring )到 MessageService
一次。对注入(inject)的原型(prototype) bean 执行的方法调用之后将不会代理到新的原型(prototype) bean 实例。
通过将proxyMode
设置为TARGET_CLASS
,ApplicationContext
不会直接在另一个bean中注入(inject)新的原型(prototype)bean实例,而是注入(inject)一个原型(prototype) bean 的代理。因此,当单例 bean 调用注入(inject)的单例 bean 中的方法时,中间代理引用新的原型(prototype) bean 并调用该方法。
更多信息可以在Spring documentation中找到:
If you want to inject (for example) an HTTP request-scoped bean into another bean of a longer-lived scope, you may choose to inject an AOP proxy in place of the scoped bean. That is, you need to inject a proxy object that exposes the same public interface as the scoped object but that can also retrieve the real target object from the relevant scope (such as an HTTP request) and delegate method calls onto the real object.
关于java - 为什么在将 CGLIB 原型(prototype)注入(inject) Singleton 的情况下,每次对原型(prototype)的访问都会创建一个新对象?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58136577/
以下代码,我使用 chrome 浏览器控制台进行了检查: function A(){ this.a='a' } 这是一个构造函数。我已经将一个属性 b 赋给了 A 的原型(prototype)。
就目前而言,这个问题不适合我们的问答形式。我们希望答案得到事实、引用资料或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
关闭。这个问题是opinion-based .它目前不接受答案。 想改进这个问题?更新问题,以便 editing this post 提供事实和引用来回答它. 5年前关闭。 Improve this
关闭。这个问题是opinion-based .它目前不接受答案。 想改进这个问题?更新问题,以便 editing this post 提供事实和引用来回答它. 3年前关闭。 Improve this
我已经开始阅读 The Pragmatic Programmer,我很喜欢并学习堆形式,但我很难理解示踪子弹和原型(prototype)之间的区别。跟踪项目符号是否像拥有应用程序的所有 View 但尚
尽管阅读了 StackOverflow 上的大多数文章,但我现在实际上对原型(prototype)非常困惑。 function Foo() { } Foo.prototype.speak = func
我正在阅读以下代码,并开始想知道 Rectangle.prototype = Object.create(Shape.prototype) 和 Rectangle.prototype = Shape.
我想知道它们之间的区别: childObj.prototype = Object.create(parentObj.prototype) 和 childObj.prototype = parentOb
这个问题在这里已经有了答案: Why wouldn't I use Child.prototype = Parent.Prototype rather than Child.prototype =
在 node.js 中导出原型(prototype)的首选方法是什么?您可以采用两种方法: 导出原型(prototype)本身 function A () { } module.exports = A
我正在学习 JavaScript,发现了两种分配原型(prototype)的方法。 第一个是A.prototype = B.prototype,第二个是A.prototype = new B() 例如
在一些构造函数的定义之后,例如 child ,我见过以下两种形式: Child.prototype = Parent.prototype; 或 Child.prototype = new Parent
我正在阅读一本关于 OOP javascript 的书,但被其中一个示例卡住了。 在示例代码的第一个版本中,Shape 的一个新实例构造函数被创建并且 toString方法被调用。 toString方
这个问题在这里已经有了答案: What should I connect to the child prototype property in JavaScript (2 个答案) 关闭 8 年前。
在进行原型(prototype)设计时,您在多大程度上放弃了最佳实践来支持代码和修复黑客攻击?当然,代码并不打算在完整的生产环境中保留。 补充:我正在研究一个用 Python 制作的相当大的半工作原型
我正在尝试使用 Prototype 更新隐藏表单字段的值。表单域: 我正在尝试使用原型(prototype)更新值: var additionalVal = ',2'; var itemId = $
我正在阅读How to Make a Javascript Library我发现了作者所说的一个观点: function _() { //Some obects and var
我想用一个新函数扩展“Number”类型,因此我必须定义一个原型(prototype)。当我想到这一点时,我得到了一堆问题: Number 是否既继承了 Object.prototype 又继承了 F
这里好像有区别... 假设我们有 function MyConstructor() {} MyConstructor 的[[Prototype]] 是Function.prototype,不是 MyC
有人建议 Derived.prototype = Object.create(Base.prototype); 优于 Derived.prototype = new Base(); (如 this S
我是一名优秀的程序员,十分优秀!