淺談Spring AOP中args()和argNames的含義
1、切入點表達式部分如果增加了args()部分,那么目標方法除了要滿足execution部分,還要滿足args()對方法參數的要求,對于符合execution表達式,但不符合args參數的方法,不會被植入切面。
2、定義了args()之后,才能把目標方法的參數傳入到切面方法的參數中(通過Joinpoint也可以獲取參數,但當前方法是直接用切面方法參數接受)。
示例1目標方法:
@RestController@RequestMapping('/testAop')public class TestController { private Logger logger = LoggerFactory.getLogger(TestController.class); @RequestMapping('/helloworld') public String helloWorld(String id, Integer age){System.out.println('被代理方法正在執行');return null; }}
切面方法
@After('execution(* com.bxp.controller.TestController.*(..)) && args(userId, userAge)') public void after(JoinPoint point, String userId, Integer userAge){System.out.println('userId===========' + userId);System.out.println('userAge===========' + userAge); }
輸出結果:
被代理方法正在執行userId===========bian1996userAge===========24
定義了args(userId, userAge)才能把目標方法helloWorld(String id, Integer age)的參數傳入到增強處理方法after的參數中,id參數對應userId,age參數對應userAge。使用的方法是按順序一一對應,helloWorld第一個參數對args第一個參數,helloWorld第2個參數對args第2個參數。
切入點表達式部分增加了&&args(userId, userAge)部分,意味著可以在增強處理方法中定義userId、userAge兩個形參------定義這兩個形參時,形參類型可以隨意指定,但是一旦指定,譬如這里分別是String類型和Integer類型,這兩個形參類型將用于限制該切入點只匹配第一個參數類型為String,第二個參數類型為Integer的方法。
也就是,args()中的參數會和目標方法的參數除了在順序上一一對應之外,在類型上也要對應,否則匹配失敗,如下兩種情況都會匹配失敗。
@RequestMapping('/helloworld')public String helloWorld(Integer id, Integer age){ System.out.println('被代理方法正在執行'); return null;} @After('execution(* com.bxp.controller.TestController.*(..)) && args(userId, userAge)')public void after(JoinPoint point, String userId, String userAge){ System.out.println('userId===========' + userId); System.out.println('userAge===========' + userAge);}@RequestMapping('/helloworld')public String helloWorld(Integer sex, String id, Integer age){ System.out.println('被代理方法正在執行'); return null;} @After('execution(* com.bxp.controller.TestController.*(..)) && args(userId, userAge)') public void after(JoinPoint point, String userId, Integer userAge){ System.out.println('userId===========' + userId); System.out.println('userAge===========' + userAge); }
除此之外,使用args()表達式時還可使用如下形式:args(userId, userAge,..),這表明增強處理方法中可以通過userId, userAge來訪問目標方法的參數。注意上面args表達式括號中的2點,它表示可以匹配更多參數,但是只要前兩個userId, userAge參數匹配上了,目標方法就可以被匹配上。
argNames是可選的,如果沒有argNames這個參數,而編譯器設置了【在class文件生成變量調試信息】,則spring可以通過反射知道方法參數的名字,通過名字配對,Spring知道args(userId, userAge)表達式里面的userId和userAge,對應了增強方法public void after(JoinPoint point, String userId, Integer userAge)方法里面的userId和userAge,就是第一個示例的情況:
總結:目標方法和args()通過參數順序一一進行匹配
args()和增強方法通過參數名稱一致進行匹配。
但是,如果設置了argNames,Spring不再使用方法參數的名字來配對,使用argNames定義的順序來給
after(JoinPoint point, String userAge, String userId)的參數傳值,例如:argNames='userId,userAge',userId在userAge前面,表示after方法第一個參數(JoinPoint 除外)是userId,第二個參數是userAge,示例如下:
目標方法
@RequestMapping('/helloworld')public String helloWorld(String id, String age){ System.out.println('被代理方法正在執行'); return null;}
切面方法
@After(value = 'execution(* com.bxp.controller.TestController.*(..)) && args(userId, userAge)', argNames = 'userId,userAge')public void after(JoinPoint point, String userAge, String userId){ System.out.println('userId===========' + userId); System.out.println('userAge===========' + userAge);}
請求連接和輸出結果
請求連接http://localhost:8088/testAop/helloworld?age=24&id=bian1996輸出結果被代理方法正在執行userId===========24userAge===========bian1996注意:這一次兩個參數的類型都給成String類型了
總結:目標方法和args()通過參數順序一一進行匹配
args()和argNames通過參數名稱一致進行匹配
argNames和增強方法通過參數順序一一對應。
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持好吧啦網。
相關文章: