作者:文岚 | 来源:互联网 | 2023-12-09 10:25
本文介绍了使用SpringAOP实现切面编程的步骤和注意事项。首先解释了@EnableAspectJAutoProxy、@Aspect、@Pointcut等注解的作用,并介绍了实现AOP功能的方法。然后详细介绍了创建切面、编写测试代码的过程,并展示了测试结果。接着讲解了关于环绕通知的使用方法,并修改了FirstTangent类以添加环绕通知方法。最后介绍了利用AOP拦截注解的方法,只需修改全局切入点即可实现。使用SpringAOP进行切面编程可以方便地实现对代码的增强和拦截。
一、注解解释
- @EnableAspectJAutoProxy: 开启springAOP代理功能
- @Aspect:把当前类标识为一个切面供容器读取
- @Pointcut:声明全局切入点,声明的方法只是该注解的载体,具体切入点只需引入载体即可
- @Before:前置通知
- @AfterReturning:后置通知
- @Around:环绕通知 即:前置通知 + 后置通知
- @After:最终通知
- @AfterThrowing:异常通知
二、实现AOP功能
org.aspectj
aspectjweaver
1.9.2
package com.test.spring.test3;
public interface TestAop {
void testAop1();
}
package com.test.spring.test3;
import org.springframework.stereotype.Component;
@Component
public class TestAopImpl implements TestAop {
@Override
public void testAop1() {
System.out.println("TestAopImpl.testAop1()");
}
}
package com.test.spring.test3.aop;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;
/**
* 需要将定义的切面类注册到springIOC容器中,否则spring无法实现aop功能(因为spring没有切面类的 实例)
*/
@Aspect
@Component
public class FirstTangent {
@Pointcut("execution(public * com.test.spring.test3.TestAopImpl.testAop1(..))")
public void url(){}
/**
* 前置通知
*/
@Before("url()")
public void testBefore(){
System.out.println("testBefore");
}
/**
* 后置通知
*/
@AfterReturning("url()")
public void testAfterReturning(){
System.out.println("testAfterReturning");
}
}
package com.test.spring.test3;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
/**
* @EnableAspectJAutoProxy --- 开启springAOP代理功能
*/
@ComponentScan
@Configuration
@EnableAspectJAutoProxy
public class AopConfig {
}
package com.test.spring.test3;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = AopConfig.class)
public class Test3 {
@Autowired
private TestAop testAop;
@Test
public void TestAop(){
testAop.testAop1();
}
}
三、关于环绕通知
编写环绕通知需要使用参数ProceedingJoinPoint,当前置通知执行完毕后使用proceed方法放行,当目标方法执行完毕后会返回到此通知方法,继续执行后置通知
@Aspect
@Component
public class FirstTangent {
@Pointcut("execution(public * com.test.spring.test3.TestAopImpl.testAop1(..))")
public void url(){}
/**
* 前置通知
*/
//@Before("url()")
public void testBefore(){
System.out.println("testBefore");
}
//@AfterReturning("url()")
public void testAfterReturning(){
System.out.println("testAfterReturning");
}
//@AfterThrowing("url()")
public void testAfterThrowing(){
System.out.println("testAfterThrowing");
}
@Around("url()")
public void testAround(ProceedingJoinPoint pjp){
//----------前置-----------
System.out.println("testAround1");
//----------执行目标方法-----------
try {
pjp.proceed();
} catch (Throwable throwable) {
throwable.printStackTrace();
}
//-------------后置-----------------
System.out.println("testAround2");
}
}
四、利用AOP拦截注解
只需修改全局切入点即可:
如:@Pointcut("@annotation(com.test.spring.multi_data_source.annotation.DataSource)")
哈哈,使用自定义注解的时候是不是方便了很多