-
Notifications
You must be signed in to change notification settings - Fork 38.3k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
AdvisedSupport.MethodCacheKey
should check for logical equality as well as identity
#33915
Comments
@bclozel Hello, the code and README in this repository provide a more detailed description of the issue. I would be very grateful if you could help confirm this problem. |
AdvisedSupport.MethodCacheKey#equals
, using ==
to determine object equality may lead to issues
AdvisedSupport.MethodCacheKey#equals
, using ==
to determine object equality may lead to issuesAdvisedSupport.MethodCacheKey#equals
should check for equality as well as identity
This commit introduces a test which verifies how many times the matches() method of a StaticMethodMatcherPointcut is invoked during ApplicationContext startup as well as during actual method invocations via the advice chain, which also indirectly tests the behavior of the equals() implementation in AdvisedSupport.MethodCacheKey. In addition, this commit revises BeanFactoryTransactionTests to assert that a transaction is started for the setAge() method. See gh-33915
Prior to this commit, the equals() implementation in AdvisedSupport's MethodCacheKey only considered methods to be equal based on an identity comparison (`==`). However, an identity comparison is not always sufficient due to the following. - Class#getDeclaredMethods() always returns "child copies" of the underlying Method instances -- which means that `equals()` should be used instead of (or in addition to) `==` whenever the compared Method instances can come from different sources. With this commit, the equals() implementation in MethodCacheKey now considers methods equal based on identity or logical equality, giving preference to the quicker identity check. See spring-projectsgh-32586 Closes spring-projectsgh-33915
AdvisedSupport.MethodCacheKey#equals
should check for equality as well as identityAdvisedSupport.MethodCacheKey
should check for logical equality as well as identity
Congratulations on submitting your first issue for the Spring Framework! 👍 And thanks for the detailed explanation and sample project. In commit 320831b, I introduced In addition, I pushed a proposed enhancement for main...sbrannen:spring-framework:issues/gh-33915-AdvisedSupport-MethodCacheKey As a side note, this is also related to: Regards, Sam |
Prior to this commit, the equals() implementation in AdvisedSupport's MethodCacheKey only considered methods to be equal based on an identity comparison (`==`). However, an identity comparison is not always sufficient due to the following. - Class#getDeclaredMethods() always returns "child copies" of the underlying Method instances -- which means that `equals()` should be used instead of (or in addition to) `==` whenever the compared Method instances can come from different sources. With this commit, the equals() implementation in MethodCacheKey now considers methods equal based on identity or logical equality, giving preference to the quicker identity check. See spring-projectsgh-32586 Closes spring-projectsgh-33915
Prior to this commit, the equals() implementation in AdvisedSupport's MethodCacheKey only considered methods to be equal based on an identity comparison (`==`), which led to duplicate entries in the method cache for the same logical method. This is caused by the fact that AdvisedSupport's getInterceptorsAndDynamicInterceptionAdvice() method is invoked at various stages with different Method instances for the same method: 1) when creating the proxy 2) when invoking the method via the proxy The reason the Method instances are different is due to the following. - Methods such as Class#getDeclaredMethods() and Class#getDeclaredMethod() always returns "child copies" of the underlying Method instances -- which means that `equals()` should be used instead of (or in addition to) `==` whenever the compared Method instances can come from different sources. With this commit, the equals() implementation in MethodCacheKey now considers methods equal based on identity or logical equality, giving preference to the quicker identity check. See gh-32586 Closes gh-33915
In the production environment, I discovered that
AdvisedSupport#methodCache
contains two "identical" keys, both corresponding to the same method object.Experimentally, I found that the
methodCache
established during bean initialization is practically useless. During actual method invocation, the interceptor chain retrieved viamethodCache.get(cacheKey)
turns out to be empty. Thus, the interceptor chain is fetched again usingadvisorChainFactory
and then placed into themethodCache
. This means that the initialization logic related togetInterceptorsAndDynamicInterceptionAdvice
will not only execute during bean initialization but also during the first method invocation.Returning to the issue itself, since each call to
getClass().getDeclaredMethods()[0]
results in aMethod
object that, when compared using==
, returnsfalse
(even if it's the same method), I believe theequals
method ofMethodCacheKey
is poorly designed.The text was updated successfully, but these errors were encountered: