I'm new to AOP and I need to use AspectJ on my project. I need to use around advice but I have a problem using it, I've the following code in my .aj class,
pointcut checkUser(ProceedingJoinPoint jp,User user): call(* com.example.UserAccount.MyUI.checkUser(..))&& args(jp,user);
void around(ProceedingJoinPoint jp,User user) throws Throwable : checkUser(jp,user){
// Condition checks one of the user boolean property
if(condition){
jp.proceed();
}else{
// Do nothing
}
}
but I get this warning all the time,
advice defined in Aspects.UserAccount has not been applied [Xlint:adviceDidNotMatch]
By the way, I tried it without ProceedingJoinPoint
and tried just proceed();
but then got this warning, too few arguments to proceed, expected 1
I'm thankful for any single help or hint!
Reza
First I recommend to read the AspectJ documentation in order to learn the syntax. As you are using native AspectJ syntax, this is like learning a new programming language or at least a Java extension. What you are doing is mix native syntax with annotation-based syntax. Try to stick with one. I am sure that you did not find this in any tutorial but ended up with that syntax via trial and error.
You do not need to bind a joinpoint parameter in native syntax because it is there implicitly and automatically. The automatically bound joinpoint is always named
thisJoinPoint
as all tutorials surely show you. Only in annotation-based syntax you need to bind the joinpoint and can name it as you wish, but even then I recommend to stick withthisJoinPoint
because then refactoring from annotation to native syntax is easier and your eyes get used to spotting that variable name in your aspect code.The warning you get means that the pointcut you defined does not match any part of your code, at least not any part which is visible to the aspect weaver or compiler. There could be plenty of reasons why this can occur, e.g. misspelled package or class names, wrong around advice return type (return type must be
Object
for non-void methods or more specifically match what the method you want to intercept returns). Assuming that e.g.checkUser(..)
returns aboolean
, the around advice should do the same. I made up an example using your package and class names. Besides, package names should be lower-case but I used yours, assuming they are really package names and not inner classes:Helper class:
Class targeted by aspect + sample main method:
As you can see, we expect an output of "true" for the first and last entry, but "false" for the ones in between due to the check logic I made up for
checkUser(..)
.Now let us write an aspect which also returns "true" for a user named "Xander", e.g. in order to give him admin rights or whatever. I am making this up because you did not provide an MCVE as you always should on StackOverflow, but just an incoherent code snippet which keeps everyone trying to answer your question guessing what the heck you might want to achieve and how to reproduce your problem.
Aspect:
I just imported the
MyUI
class, so there is no need to use a fully-qualified class name here. Again, this is an advantage of native syntax, in annotation-based syntax you would have to use the fully qualified name.I also replaced the generic
* MyUI.checkUser(..)
(which would also work) by the more explicitboolean MyUI.checkUser(*)
because we already know that the method returns a boolean and has exactly one parameter, which both we assume anyway by returning a boolean from the around advice and by binding exactly one parameter viaargs()
. You could also be even more specific and useboolean MyUI.checkUser(User)
.Furthermore, I am using
execution()
rather thancall()
because it is more efficient, as it weaves the advice code just into the executing method once instead of five times for each method call in the main method. You only need to usecall()
if theMyUI
class is out of reach of the AspectJ weaver/compiler, i.e. because it is not in the module you compile with AspectJ Maven.Console log:
Et voilà, the aspect works. It makes the target method return "true" for user "xander".