I am using AspectJ in Java to log the calls to some methods. I've looked online but couldn't manage to find an answer to this:
What happens when two @Around
advices match for a method?
Specifically, I am using two @Around advices, like this:
@Around("condition1() && condition2() && condition3()")
public Object around(ProceedingJoinPoint point) {
return around(point, null);
}
@Around("condition1() && condition2() && condition3() && args(request)")
public Object around(ProceedingJoinPoint point, Object request) {
...
result = (Result) point.proceed();
...
}
Will this result in point.proceed()
being called twice (having the actual method called twice) if both of these advices match?
Your approach is highly problematic because you manually call one advice from another one. This is not how AOP should be applied. Please let AspectJ decide which advices to execute based on their respective pointcuts. The way you delegate from one advice to another you could even call an advice which would not match by itself. Example in plain AspectJ without Spring (works the same in Spring AOP, though):
Java driver application:
Aspect:
Console log:
Can you see how unhealthy your approach is? An advice which otherwise would not match is called by a matching one. This yields some really unexpected behaviour IMO. Please don't do it!!!
Now as for your original question about multiple matching advice, this is how you should do it:
Modified aspect:
New console log:
As you can see, AspectJ or Spring AOP wrap multiple matching advice like onion skins around joinpoints and only the innermost
proceed()
calls the actual joinpoint while the outer layers call the inner ones, making sure that each joinpoint is executed only once. There is no need for you trying to be smarter than the AOP framework, possibly causing damage (see my first example).One more thing: If multiple aspects have matching pointcuts, you can influence their order of execution via
@DeclarePrecedence
in AspectJ, but within a single aspect you have no influence on the execution order or at least you should not rely on it. In Spring AOP you can use the@Order
annotation in order to determine aspect precedence, but the order is also undefined for multiple advice from the same aspect, see also the Spring manual.Update 2016-02-28, 18:30 CET, after some discussion in comments:
Okay, we extend the driver class a little bit so we can test some more:
Now, binding the first parameter in AspectJ is as easy as
args(request, ..)
which works for one or more parameters. The only exception is zero parameters, in which case the pointcut would not fire. So either I end up with something similar to what you did:Which is makes the same advice fire twice and thus causes an overhead, even though the original method is only called once, but you can see the overhead in the log:
You can easily recognise how double advices are fired for each joinpoint.
Alternatively, you can bind the parameter during runtime, which is not very elegant and incurs a little runtime penalty, but works perfectly well:
This avoids double advice execution as well as code duplication and yields the following console output:
Last, but not least, you can have two slightly different pointcuts - one with empty
args()
and one withargs(request, ..)
- both of which can delegate parameter handling, logging and exception handling to a helper method in order to avoid duplication, as I said in one of my comments:The console log should be exactly the same as the previous one.
Update 2:
Well, I just realised that the empty
args()
trick would also apply to your original idea and avoid double execution as well as the helper method:This is acceptable as well as elegant because it does not generate byte code twice per joinpoint. The two pointcuts are mutually exclusive, so this is a good thing to do. I recommend this solution.