- 两种方式实现AOP
基于XML配置开发AspectJ
package aspectj.xml;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
public class MyAspect {
/**
* 切面类,在此类中编写各种类型通知
*/
public void before(JoinPoint jp) {
System.out.print("前置通知:模拟权限控制");
System.out.println(",目标类对象:" + jp.getTarget()
+ ",被增强处理的方法:" + jp.getSignature().getName());
}
/**
* 前置通知,使用Joinpoint接口作为参数获得目标对象信息
*/
public void afterReturning(JoinPoint jp) {
System.out.print("后置返回通知:" + "模拟删除临时文件");
System.out.println(",被增强处理的方法:" + jp.getSignature().getName());
}
/**
* 环绕通知
* ProceedingJoinPoint是JoinPoint子接口,代表可以执行的目标方法
* 返回值类型必须是Object
* 必须一个参数是ProceedingJoinPoint类型
* 必须throws Throwable
*/
public Object around(ProceedingJoinPoint pjp) throws Throwable{
//开始
System.out.println("环绕开始:执行目标方法前,模拟开启事务");
//执行当前目标方法
Object obj = pjp.proceed();
//结束
System.out.println("环绕结束:执行目标方法后,模拟关闭事务");
return obj;
}
/**
* 异常通知
*/
public void except(Throwable e) {
System.out.println("异常通知:" + "程序执行异常" + e.getMessage());
}
/**
* 后置(最终)通知
*/
public void after() {
System.out.println("最终通知:模拟释放资源");
}
}
<?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.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd">
<!-- 定义目标对象,使用4.2.1节的实现类 -->
<bean id="testDao" class="dynamic.jdk.TestDaoImpl"/>
<!-- 定义切面 -->
<bean id="myAspect" class="aspectj.xml.MyAspect"/>
<!-- AOP配置 -->
<aop:config>
<!-- 配置切面 -->
<aop:aspect ref="myAspect">
<!-- 配置切入点,通知增强哪些方法 -->
<aop:pointcut expression="execution(* dynamic.jdk.*.*(..))" id="myPointCut"/>
<!-- 将通知与切入点关联 -->
<!-- 关联前置通知 -->
<aop:before method="before" pointcut-ref="myPointCut"/>
<!-- 关联后置返回通知,在目标方法成功执行后执行 -->
<aop:after-returning method="afterReturning" pointcut-ref="myPointCut"/>
<!-- 关联环绕通知 -->
<aop:around method="around" pointcut-ref="myPointCut"/>
<!-- 关联异常通知,没有异常发生时,将不会执行增强,throwing属性设置通知的第二个参数名称 -->
<aop:after-throwing method="except" pointcut-ref="myPointCut" throwing="e"/>
<!-- 关联后置(最终)通知,不管目标方法是否成功,都要执行 -->
<aop:after method="after" pointcut-ref="myPointCut"/>
</aop:aspect>
</aop:config>
</beans>
package aspectj.xml;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import dynamic.jdk.TestDao;
public class XMLAspectJTest {
public static void main(String[] args) {
ApplicationContext appCon = new ClassPathXmlApplicationContext("/aspectj/xml/applicationContext.xml");
//从容器中,获取增强后的目标对象
TestDao testDaoAdvice = (TestDao)appCon.getBean("testDao");
//执行方法
testDaoAdvice.save();
}
}
基于注解开发AspectJ
package aspectj.annotation;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;
/**
* 切面类,在此类中编写各种类型通知
*/
@Aspect//对应<aop:aspect ref="myAspect">
@Component//对应<bean id="myAspect" class="aspectj.xml.MyAspect"/>
public class MyAspect {
/**
* 定义切入点
*/
@Pointcut("execution(* dynamic.jdk.*.*(..))")
private void myPointCut() {
//对应<aop:pointcut expression="execution(* dynamic.jdk.*.*(..))" id="myPointCut"/>
}
/**
* 前置通知,使用Joinpoint接口作为参数获得目标对象信息
*/
@Before("myPointCut()")//对应<aop:before method="before" pointcut-ref="myPointCut"/>
public void before(JoinPoint jp) {
System.out.print("前置通知:模拟权限控制");
System.out.println(",目标类对象:" + jp.getTarget()
+ ",被增强处理的方法:" + jp.getSignature().getName());
}
/**
* 后置返回通知
*/
@AfterReturning("myPointCut()")
public void afterReturning(JoinPoint jp) {
System.out.print("后置返回通知:" + "模拟删除临时文件");
System.out.println(",被增强处理的方法:" + jp.getSignature().getName());
}
/**
* 环绕通知
* ProceedingJoinPoint是JoinPoint子接口,代表可以执行的目标方法
* 返回值类型必须是Object
* 必须一个参数是ProceedingJoinPoint类型
* 必须throws Throwable
*/
@Around("myPointCut()")
public Object around(ProceedingJoinPoint pjp) throws Throwable{
//开始
System.out.println("环绕开始:执行目标方法前,模拟开启事务");
//执行当前目标方法
Object obj = pjp.proceed();
//结束
System.out.println("环绕结束:执行目标方法后,模拟关闭事务");
return obj;
}
/**
* 异常通知
*/
@AfterThrowing(value="myPointCut()",throwing="e")
public void except(Throwable e) {
System.out.println("异常通知:" + "程序执行异常" + e.getMessage());
}
/**
* 后置(最终)通知
*/
@After("myPointCut()")
public void after() {
System.out.println("最终通知:模拟释放资源");
}
}
package aspectj.annotation;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import dynamic.jdk.TestDao;
public class AnnotationAspectJTest {
public static void main(String[] args) {
ApplicationContext appCon = new ClassPathXmlApplicationContext("/aspectj/xml/applicationContext.xml");
//从容器中,获取增强后的目标对象
TestDao testDaoAdvice = (TestDao)appCon.getBean("testDao");
//执行方法
testDaoAdvice.save();
}
}
<?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"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd">
<!-- 指定需要扫描的包,使注解生效 -->
<context:component-scan base-package="aspectj.annotation"/>
<context:component-scan base-package="dynamic.jdk"/>
<!-- 启动基于注解的AspectJ支持 -->
<aop:aspectj-autoproxy/>
</beans>
五种通知类型:环绕、前置、后置返回、后置、异常通知
八个概念:切面、连接点、切入点、通知、引入、目标对象、代理、织入