亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

基于Spring AOP @AspectJ進(jìn)階說明

 更新時間:2021年01月20日 11:35:48   作者:我是一名老菜鳥  
這篇文章主要介紹了基于Spring AOP @AspectJ進(jìn)階說明,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧

@AspectJ可以使用切點函數(shù)定義切點,我們還可以使用邏輯運算符對切點進(jìn)行復(fù)核運算得到復(fù)合的切點,為了在切面中重用切點,我們還可以對切點進(jìn)行命名,以便在其他的地方引用定義過的切點。

當(dāng)一個連接點匹配多個切點時,需要考慮織入順序的問題,此外一個重要的問題是如何再增強(qiáng)中訪問連接點上下文的信息。

Waiter接口:

package com.yyq.aspectJAdvanced;
public interface Waiter {
 void greetTo(String name);
 void serveTo(String name);
}

NaiveWaiter實現(xiàn)類:

package com.yyq.aspectJAdvanced;
public class NaiveWaiter implements Waiter {
 @Override
 public void greetTo(String name) {
  System.out.println("NaiveWaiter:greet to " + name + "...");
 }
 @Override
 public void serveTo(String name) {
  System.out.println("NaiveWaiter:serving to " + name + "...");
 }
 public void smile(String clientName,int times){
  System.out.println("NaiveWaiter:smile to "+clientName+ times+"times...");
 }
}

NaughtyWaiter實現(xiàn)類:

package com.yyq.aspectJAdvanced;
public class NaughtyWaiter implements Waiter {
 public void greetTo(String clientName) {
  System.out.println("NaughtyWaiter:greet to " + clientName + "...");
 }
 public void serveTo(String clientName) {
  System.out.println("NaughtyWaiter:serving " + clientName + "...");
 }
 public void joke(String clientName, int times) {
  System.out.println("NaughtyWaiter:play " + times + " jokes to " + clientName + "...");
 }
}

Seller接口:

package com.yyq.aspectJAdvanced;
public interface Seller {
 int sell(String goods, String clientName);
}

SmallSeller實現(xiàn)類:

package com.yyq.aspectJAdvanced;
public class SmartSeller implements Seller {
 public int sell(String goods,String clientName) {
  System.out.println("SmartSeller: sell "+goods +" to "+clientName+"...");
  return 100;
 }
 
 public void checkBill(int billId){
  if(billId == 1) throw new IllegalArgumentException("iae Exception");
  else throw new RuntimeException("re Exception");
 }
}

beans.xml配置文件:

<?xml version="1.0" encoding="UTF-8" ?>
<beans xmlns="http://www.springframework.org/schema/beans"
  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  xmlns:aop="http://www.springframework.org/schema/aop"
  xsi:schemaLocation="http://www.springframework.org/schema/beans
  http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
  http://www.springframework.org/schema/aop
  http://www.springframework.org/schema/aop/spring-aop-3.0.xsd">
 <aop:aspectj-autoproxy proxy-target-class="true"/>
 <bean id="naiveWaiter" class="com.yyq.aspectJAdvanced.NaiveWaiter"/>
 <bean id="naughtyWaiter" class="com.yyq.aspectJAdvanced.NaughtyWaiter"/>
 <bean id="seller" class="com.yyq.aspectJAdvanced.SmartSeller"/>
 <!--
 <bean class="com.yyq.aspectJAdvanced.TestAspect"/>
 
 <bean class="com.yyq.aspectJAdvanced.TestAspect2"/>
 <bean class="com.yyq.aspectJAdvanced.TestAspect3"/>
 <bean class="com.yyq.aspectJAdvanced.TestAspect4"/>
 <bean class="com.yyq.aspectJAdvanced.TestAspect5"/>
 <bean id="naiveWaiter2" class="com.yyq.aspectJAdvanced.NaiveWaiter2"/>
 <bean class="com.yyq.aspectJAdvanced.TestAspect6"/>
 <bean class="com.yyq.aspectJAdvanced.TestAspect7"/>
 <bean class="com.yyq.aspectJAdvanced.TestAspect8"/>
-->
</beans>

1、切點符合運算

使用切點符合運算符,我們將擁有強(qiáng)大而靈活的切點表達(dá)能力。

TestAspect:切點符合運算定義切面

package com.yyq.aspectJAdvanced;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
@Aspect
public class TestAspect {
 //與非運算
 @Before("!target(com.yyq.aspectJAdvanced.NaiveWaiter) && execution(* serveTo(..))")
 public void notServeInNaiveWaiter(){
  System.out.println("--notServeInNaiveWaiter() executed!--");
 }
 //與運算
 @After("within(com.yyq.aspectJAdvanced.*) && execution(* greetTo(..))")
 public void greetToFun(){
  System.out.println("--greetToFun() executed!--");
 }
 //或運算
 @AfterReturning("target(com.yyq.aspectJAdvanced.Waiter) || target(com.yyq.aspectJAdvanced.Seller)")
 public void waiterOrSeller(){
  System.out.println("--waiterOrSeller() executed!--");
 }
}

測試方法:

@Test
 public void pointAspectJTest() {
  String configPath = "com\\yyq\\aspectJAdvanced\\beans.xml";
  ApplicationContext ctx = new ClassPathXmlApplicationContext(configPath);
  Waiter naiveWaiter = (Waiter) ctx.getBean("naiveWaiter");
  Waiter naughtyWaiter = (Waiter) ctx.getBean("naughtyWaiter");
  naiveWaiter.greetTo("John");
  naiveWaiter.serveTo("John");
  naughtyWaiter.greetTo("Tom");
  naughtyWaiter.serveTo("Tom");
 }

輸出結(jié)果:

NaiveWaiter:greet to John...
--greetToFun() executed!--
--waiterOrSeller() executed!--
NaiveWaiter:serving to John...
--waiterOrSeller() executed!--
NaughtyWaiter:greet to Tom...
--greetToFun() executed!--
--waiterOrSeller() executed!--
--notServeInNaiveWaiter() executed!--
NaughtyWaiter:serving Tom...
--waiterOrSeller() executed!--

2、命名切點

切點直接聲明在增強(qiáng)方法處被稱為匿名切點,匿名切點只能在聲明處使用。如果希望在其他地方重用一個切點,我們可以通過@Pointcut注解以及切面類方法對切點進(jìn)行命名。

TestNamePointcut:命名切點類

package com.yyq.aspectJAdvanced;
import org.aspectj.lang.annotation.Pointcut;
public class TestNamePointcut {
 //通過注解方法inPackage()對該切點進(jìn)行命名,方法可視域修飾符為private,表明該命名切點只能在本切面類中使用
 @Pointcut("within(com.yyq.aspectJAdvaned.*)")
 private void inPackage(){}
 @Pointcut("execution(* greetTo(..))")
 protected void greetTo(){}
 @Pointcut("inPackage() and greetTo()")
 public void inPkgGreetTo(){}
}

TestAspect2:切面實現(xiàn)類

package com.yyq.aspectJAdvanced;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
@Aspect
public class TestAspect2 {
 @Before("TestNamePointcut.inPkgGreetTo()")
 public void pkgGreetTo(){
  System.out.println("--pkgGreetTo() executed!--");
 }
 @Before("target(com.yyq.aspectJAdvanced.NaiveWaiter) || TestNamePointcut.inPkgGreetTo()")
 public void pkgGreetToNotnaiveWaiter(){
  System.out.println("--pkgGreetToNotnaiveWaiter() executed!--");
 }
}

測試方法:

@Test
 public void pointAspectJTest2() {
  String configPath = "com\\yyq\\aspectJAdvanced\\beans.xml";
  ApplicationContext ctx = new ClassPathXmlApplicationContext(configPath);
  NaiveWaiter naiveWaiter = (NaiveWaiter) ctx.getBean("naiveWaiter");
  naiveWaiter.smile("Andy", 2);
 }

輸出結(jié)果:

--pkgGreetToNotnaiveWaiter() executed!--
NaiveWaiter:smile to Andy2times...

3、增強(qiáng)織入的順序

一個連接點可以同時匹配多個切點,切點對應(yīng)的增強(qiáng)在連接點上的織入順序的安排主要有以下3種情況:

1)如果增強(qiáng)在同一個切面類中聲明,則依照增強(qiáng)在切面類中定義的順序進(jìn)行織入;

2)如何增強(qiáng)位于不同的切面類中,且這些切面類都實現(xiàn)了org.springframework.core.Order接口,則由接口方法的順序號決定(順序號小的先織入);

3)如果增強(qiáng)位于不同的切面類中,且這些切面類沒有實現(xiàn)org.springframework.core.Order接口,織入的順序是不確定的。

4、訪問連接點信息

AspectJ使用org.aspectj.lang.JoinPoint接口表示目標(biāo)類連接點對象,如果是環(huán)繞增強(qiáng)時,使用org.aspectj.lang.ProceedingJoinPoint表示連接點對象,該類是JoinPoint的子接口,任何一個增強(qiáng)方法都可以通過將第一個入?yún)⒙暶鳛镴oinPoint訪問到連接點上下文的信息。

TestAspect3:切面實現(xiàn)類

@Aspect
public class TestAspect3 {
 @Around("execution(* greetTo(..)) && target(com.yyq.aspectJAdvanced.NaiveWaiter)")
 public void joinPointAccess(ProceedingJoinPoint pjp) throws Throwable {
  System.out.println("---joinPointAccess---");
  System.out.println("args[0]:" + pjp.getArgs()[0]);
  System.out.println("signature:" + pjp.getTarget().getClass());
  pjp.proceed();
  System.out.println("---joinPointAccess---");
 }
}

測試方法:

 @Test
 public void pointAspectJTest3() {
  String configPath = "com\\yyq\\aspectJAdvanced\\beans.xml";
  ApplicationContext ctx = new ClassPathXmlApplicationContext(configPath);
  Waiter naiveWaiter = (Waiter) ctx.getBean("naiveWaiter");
  naiveWaiter.greetTo("Andy");
 }

輸出結(jié)果:

---joinPointAccess---
args[0]:Andy
signature:class com.yyq.aspectJAdvanced.NaiveWaiter
NaiveWaiter:greet to Andy...
---joinPointAccess---

5、綁定連接點方法入?yún)?/h3>

args()用于綁定連接點方法的入?yún)ⅲ籃annotation()用于綁定連接點方法的注解對象;而@args()用于綁定連接點方法入?yún)⒌淖⒔狻?/p>

TestAspect4:切面實現(xiàn)類

@Aspect
public class TestAspect4 {
 @Before("target(com.yyq.aspectJAdvanced.NaiveWaiter) && args(name,num,..)")
 public void bindJoinPointParams(int num, String name) {
  System.out.println("---bindJoinPointParams---");
  System.out.println("name:" + name);
  System.out.println("num:" + num);
  System.out.println("---bindJoinPointParams---");
 }
}

測試方法:

@Test
 public void pointAspectJTest4() {
  String configPath = "com\\yyq\\aspectJAdvanced\\beans.xml";
  ApplicationContext ctx = new ClassPathXmlApplicationContext(configPath);
  NaiveWaiter naiveWaiter = (NaiveWaiter) ctx.getBean("naiveWaiter");
  naiveWaiter.smile("Andy", 3);
 }

輸出結(jié)果:

---bindJoinPointParams---
name:Andy
num:3
---bindJoinPointParams---
NaiveWaiter:smile to Andy 3 times...

6、綁定代理對象

使用this()或target()可綁定被代理對象實例,在通過類實例名綁定對象時,還依然具有原來連接點匹配的功能,只不過類名是通過增強(qiáng)方法中同名入?yún)⒌念愋烷g接決定罷了。

TestAspect5:切面實現(xiàn)類

@Aspect
public class TestAspect5 {
 @Before("this(waiter)")
 public void bindProxyObj(Waiter waiter){
  System.out.println("---bindProxyObj---");
  System.out.println(waiter.getClass().getName());
  System.out.println("---bindProxyObj---");
 }
}

測試方法:

@Test
 public void pointAspectJTest5() {
  String configPath = "com\\yyq\\aspectJAdvanced\\beans.xml";
  ApplicationContext ctx = new ClassPathXmlApplicationContext(configPath);
  Waiter waiter = (Waiter) ctx.getBean("naiveWaiter");
  waiter.greetTo("Yang");
 }

輸出結(jié)果:

---bindProxyObj---
com.yyq.aspectJAdvanced.NaiveWaiter$$EnhancerByCGLIB$$fefafe52
---bindProxyObj---
NaiveWaiter:greet to Yang...

7、綁定類注解對象

@within()和@target()函數(shù)可以將目標(biāo)類的注解對象綁定到增強(qiáng)方法中,我們通過@within()演示注解綁定的操作。

TestAspect6:切面測試類

@Aspect
public class TestAspect6 {
 @Before("@within(m)")
 public void bindTypeAnnoObject(Monitorable m) {
  System.out.println("---bindTypeAnnoObject---");
  System.out.println(m.getClass().getName());
  System.out.println("---bindTypeAnnoObject---");
 }
}

測試方法:

@Test
 public void pointAspectJTest6() {
  String configPath = "com\\yyq\\aspectJAdvanced\\beans.xml";
  ApplicationContext ctx = new ClassPathXmlApplicationContext(configPath);
  Waiter waiter = (Waiter) ctx.getBean("naiveWaiter2");
  ((NaiveWaiter2)waiter).greetTo("Yang");
 }

輸出結(jié)果:

---bindTypeAnnoObject---
$Proxy4
---bindTypeAnnoObject---
NaiveWaiter:greet to Yang...

8、綁定返回值

在后置增強(qiáng)中,我們可以通過returning綁定連接點方法的返回值。

TestAspect7:切面實現(xiàn)類

@Aspect
public class TestAspect7 {
 @AfterReturning(value = "target(com.yyq.aspectJAdvanced.SmartSeller)", returning = "retVal")
 public void bindReturnValue(int retVal) {
  System.out.println("---bindReturnValue---");
  System.out.println("returnValue:" + retVal);
  System.out.println("---bindReturnValue---");
 }
}

測試方法:

 @Test
 public void pointAspectJTest7() {
  String configPath = "com\\yyq\\aspectJAdvanced\\beans.xml";
  ApplicationContext ctx = new ClassPathXmlApplicationContext(configPath);
  SmartSeller seller = (SmartSeller) ctx.getBean("seller");
  seller.sell("Beer", "John");
 } 

輸出結(jié)果:

SmartSeller: sell Beer to John...
---bindReturnValue---
returnValue:100
---bindReturnValue---

9、綁定拋出的異常

和通過切點函數(shù)綁定連接點信息不同,連接點拋出的異常必須使用AfterThrowing注解的throwing成員進(jìn)行綁定。

TestAspect8:切面實現(xiàn)類

@Aspect
public class TestAspect8 {
 @AfterThrowing(value = "target(com.yyq.aspectJAdvanced.SmartSeller)", throwing = "iae")
 public void bindException(IllegalArgumentException iae) {
  System.out.println("---bindException---");
  System.out.println("exception:" + iae.getMessage());
  System.out.println("---bindException---");
 }
}

測試方法:

 @Test
 public void pointAspectJTest8() {
  String configPath = "com\\yyq\\aspectJAdvanced\\beans.xml";
  ApplicationContext ctx = new ClassPathXmlApplicationContext(configPath);
  SmartSeller seller = (SmartSeller) ctx.getBean("seller");
  seller.checkBill(1);
 }

輸出結(jié)果:

---bindException---
exception:iae Exception
---bindException---

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。

相關(guān)文章

  • mybatis-plus中的Enum用法實例

    mybatis-plus中的Enum用法實例

    本文主要介紹了mybatis-plus中的Enum用法實例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-01-01
  • SpringBoot?docker項目部署實戰(zhàn)

    SpringBoot?docker項目部署實戰(zhàn)

    本文主要介紹了SpringBoot?docker項目部署實戰(zhàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-08-08
  • springboot使用hibernate validator校驗方式

    springboot使用hibernate validator校驗方式

    hibernate validator提供了一套比較完善、便捷的驗證實現(xiàn)方式。下面小編給大家介紹下springboot使用hibernate validator校驗方式,感興趣的朋友一起看看吧
    2018-01-01
  • Java底層基于二叉搜索樹實現(xiàn)集合和映射/集合Set功能詳解

    Java底層基于二叉搜索樹實現(xiàn)集合和映射/集合Set功能詳解

    這篇文章主要介紹了Java底層基于二叉搜索樹實現(xiàn)集合和映射/集合Set功能,結(jié)合實例形式分析了Java使用二叉搜索樹實現(xiàn)集合和映射相關(guān)操作技巧,需要的朋友可以參考下
    2020-03-03
  • Java C++ leetcode執(zhí)行一次字符串交換能否使兩個字符串相等

    Java C++ leetcode執(zhí)行一次字符串交換能否使兩個字符串相等

    這篇文章主要為大家介紹了Java C++ leetcode1790執(zhí)行一次字符串交換能否使兩個字符串相等,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-10-10
  • springboot:接收date類型的參數(shù)方式

    springboot:接收date類型的參數(shù)方式

    這篇文章主要介紹了springboot:接收date類型的參數(shù)方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • Spring如何通過注解存儲和讀取對象詳解

    Spring如何通過注解存儲和讀取對象詳解

    在Spring中,要想更簡單的存儲和讀取對象的核心是使用注解,這篇文章主要給大家介紹了關(guān)于Spring如何通過注解存儲和讀取對象的相關(guān)資料,文中通過實例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2022-07-07
  • Java中的CountDownLatch源碼解析

    Java中的CountDownLatch源碼解析

    這篇文章主要介紹了Java中的CountDownLatch源碼解析,CountDownLatch類是一個同步輔助裝置,允許一個或多個線程去等待直到另外的線程完成了一組操作,需要的朋友可以參考下
    2023-12-12
  • java json 省市級聯(lián)實例代碼

    java json 省市級聯(lián)實例代碼

    這篇文章介紹了java json 省市級聯(lián)實例代碼,有需要的朋友可以參考一下
    2013-09-09
  • 深入了解JAVA泛型

    深入了解JAVA泛型

    這篇文章主要介紹了JAVA泛型的相關(guān)知識,文中代碼非常詳細(xì),供大家參考和學(xué)習(xí),感興趣的朋友可以了解下
    2020-06-06

最新評論