Spring AOP中@Pointcut的用法,注解開發AOP


首先在xml中開啟注解支持
<!--開啟AOP的注解支持-->
<aop:aspectj-autoproxy/>

舉例,環繞通知

創建一個切點類, com.spring.service下及其子包所有方法進行匹配
public class AopPointcutClass {
    @Pointcut("execution(* com.spring.service..*(..))") public void logsMean(){}
}

創建切面

 
          
@Component("logAopUtils")
@Aspect
public class LogAopUtils {
 @Before("com.spring.aoppackage.AopPointcutClass.logsMean()") public void beforePrintLog() { System.out.println("方法執行之前,輸出日志"); } @AfterReturning("com.spring.aoppackage.AopPointcutClass.logsMean()") public void afterReturningPrintLog() { System.out.println("方法執行之后,輸出日志"); } @AfterThrowing("com.spring.aoppackage.AopPointcutClass.logsMean()") public void afterThrowingPrintLog() { System.out.println("方法執行過程出現了異常,輸出日志"); } @After("com.spring.aoppackage.AopPointcutClass.logsMean()") public void afterPrintLog() { System.out.println("方法執行最終,輸出日志"); } }

舉例,連接點

創建一個切點類, com.spring.service下及其子包所有方法進行匹配
public class AopPointcutClass {
    @Pointcut("execution(* com.spring.service..*(..))") public void logsMean(){} }
創建切面連接點
@Component("logAopUtils")
@Aspect
public class LogAopUtils {
    @Around("com.spring.aoppackage.AopPointcutClass.logsMean()")
    public void aroundPrint(ProceedingJoinPoint joinPoint) {
        try {
            //前置通知
            System.out.println("方法執行之前");
            joinPoint.proceed(joinPoint.getArgs());//作用,執行被攔截的方法。類似於method.invoke()
            //后置通知
            System.out.println("方法執行之后");
        } catch (Throwable throwable) {
            throwable.printStackTrace();
            //異常通知
            System.out.println("方法出異常了");
        } finally {
            //最終通知
            System.out.println("方法執行最終節點");
        }
    }
}

詳解

在Spring 2.0中,Pointcut的定義包括兩個部分:Pointcut表示式(expression)Point簽名(signature)
//Pointcut表示式
@Pointcut("execution(* com.savage.aop.MessageSender.*(..))")
//Point簽名
private void log(){} 
然后要使用所定義的Pointcut時,可以指定Pointcut簽名
如下:
@Before("log()")

這種使用方式等同於以下方式,直接定義execution表達式使用

@Before("execution(* com.savage.aop.MessageSender.*(..))")

Pointcut定義時,還可以使用&&、||、! 這三個運算

@Pointcut("execution(* com.savage.aop.MessageSender.*(..))")
private void logSender(){}

@Pointcut("execution(* com.savage.aop.MessageReceiver.*(..))")
private void logReceiver(){}

@Pointcut("logSender() || logReceiver()")
private void logMessage(){}

這個例子中,logMessage()將匹配任何MessageSender和MessageReceiver中的任何方法。


還可以將一些公用的Pointcut放到一個類中,以供整個應用程序使用,如下:

public class Pointcuts {
  @Pointcut("execution(* *Message(..))")
  public void logMessage(){}

  @Pointcut("execution(* *Attachment(..))")
  public void logAttachment(){}

  @Pointcut("execution(* *Service.*(..))")
  public void auth(){}
}

在使用上面定義Pointcut時,指定完整的類名加上Pointcut簽名就可以了,如:

@Aspect
public class LogBeforeAdvice {
  @Before("com.sagage.aop.Pointcuts.logMessage()")
  public void before(JoinPoint joinPoint) {
    System.out.println("Logging before " + joinPoint.getSignature().getName());
  }
}

當基於XML Sechma實現Advice時,如果Pointcut需要被重用,可以使用<aop:pointcut></aop:pointcut>來聲明Pointcut,然后在需要使用這個Pointcut的地方,用pointcut-ref引用就行了,如:

<aop:config>
  <aop:pointcut id="log" expression="execution(* com.savage.simplespring.bean.MessageSender.*(..))"/>
  <aop:aspect id="logging" ref="logBeforeAdvice">
    <aop:before pointcut-ref="log" method="before"/>
    <aop:after-returning pointcut-ref="log" method="afterReturning"/>
  </aop:aspect>
</aop:config>
 
         

另外,除了execution表示式外,還有within、this、target、args等Pointcut表示式


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM