什么是aop
1:AOP:Aspect Oriented Programming 的缩写,意思为面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。
2:主要的功能是:日志记录,性能统计,安全控制,事务处理,异常处理等等。
AOP的实现方式
1预编译(Aspectj)
2运行期动态代理(JDK的动态代理,CGLIB动态代理)(SpringAOP,JBoosAOP)
data:image/s3,"s3://crabby-images/59bc7/59bc7bf8ef5ff84960a31723307e15db4b0b5f54" alt=""
通知的类型:
data:image/s3,"s3://crabby-images/908ad/908add008bf589a2ddf9727372a54309e29bafcf" alt=""
Spring框架中AOP的作用
1:提供了声明式的企业服务,特别是EJB的替代服务的声明
2:允许用户定制自己的方面,已完成oop与Aop的互补作用
image.png
image.png
基于配置的AOP实现
data:image/s3,"s3://crabby-images/f721c/f721c92bb59002204c226e72448f7095d0c1bb0f" alt=""
声明一个切面
data:image/s3,"s3://crabby-images/7f35c/7f35ceb6556646cc57c66768cbc4cb014f1dc5b5" alt=""
上图的意思为把id为aBean的bean作为一个切面声明,切面的id为myAspect
声明一个切入点
(springAOP和Aspectj都支持的)
data:image/s3,"s3://crabby-images/787b4/787b491315f7f70e69e4712e0f2f6149bafe02b2" alt=""
(springAOP支持的)
data:image/s3,"s3://crabby-images/b1cc0/b1cc03f00460ff373b30cd689745229e38404dec" alt=""
等等其他的,具体使用的时候可以查阅相关的切入点表达式
例子:
<bean id="moocAspect" class="com.imooc.aop.schema.advice.MoocAspect"></bean>
<bean id="aspectBiz" class="com.imooc.aop.schema.advice.biz.AspectBiz"></bean>
<aop:config>
<aop:aspect id="moocAspectAOP" ref="moocAspect">
<aop:pointcut expression="execution(* com.imooc.aop.schema.advice.biz.*Biz.*(..))" id="moocPiontcut"/>
</aop:aspect>
</aop:config>
通知advice
前置通知
data:image/s3,"s3://crabby-images/24033/2403305cf3df545e7ac039a2ecd21f6e37b0062d" alt=""
返回后通知
data:image/s3,"s3://crabby-images/081a8/081a8eb9ab8309a51144a26a7d5e5d4573aea38a" alt=""
异常后通知
data:image/s3,"s3://crabby-images/f1cdb/f1cdbd1ce66ab775edb66c13abee5f5b70ab257c" alt=""
最后通知(无论有无异常,最后都会执行)
data:image/s3,"s3://crabby-images/b1e65/b1e65845a649b5450bff4ded8641ff10faaec62f" alt=""
环绕通知
data:image/s3,"s3://crabby-images/ece59/ece590f7c1085aaa6df07363506ea7ca5ac7ced1" alt=""
环绕通知携带参数
data:image/s3,"s3://crabby-images/3a1d4/3a1d4ce412db432000dead6d73248ff13da478c6" alt=""
data:image/s3,"s3://crabby-images/f347e/f347e6d6f08ab9b1350de52be0dbba916ccd2330" alt=""
<bean id="moocAspect" class="com.imooc.aop.schema.advice.MoocAspect"></bean>
<bean id="aspectBiz" class="com.imooc.aop.schema.advice.biz.AspectBiz"></bean>
<aop:config>
<aop:aspect id="moocAspectAOP" ref="moocAspect">
<aop:pointcut expression="execution(* com.imooc.aop.schema.advice.biz.*Biz.*(..))" id="moocPiontcut"/>切入点表达式
//<aop:before method="before" pointcut-ref="moocPiontcut"/>前置通知,方法为before
//<aop:after-returning method="afterReturning" pointcut-ref="moocPiontcut"/>返回后通知
//<aop:after-throwing method="afterThrowing" pointcut-ref="moocPiontcut"/> 异常后通知
//<aop:after method="after" pointcut-ref="moocPiontcut"/> 最后通知
//<aop:around method="around" pointcut-ref="moocPiontcut"/> 环绕通知
//<aop:around method="aroundInit" pointcut="execution(* com.imooc.aop.schema.advice.biz.AspectBiz.init(String, int))
and args(bizName, times)"/>
//<aop:declare-parents types-matching="com.imooc.aop.schema.advice.biz.*(+)"
implement-interface="com.imooc.aop.schema.advice.Fit"
default-impl="com.imooc.aop.schema.advice.FitImpl"/>
</aop:aspect>
</aop:config>
环绕通知:
public Object around(ProceedingJoinPoint pjp) {
Object obj = null;
try {
System.out.println("MoocAspect around 1.");执行前操作
obj = pjp.proceed();方法
System.out.println("MoocAspect around 2.");执行后操作
} catch (Throwable e) {
e.printStackTrace();
}
return obj;
}
环绕通知携带参数:
public Object aroundInit(ProceedingJoinPoint pjp, String bizName, int times) {
System.out.println(bizName + " " + times);
Object obj = null;
try {
System.out.println("MoocAspect aroundInit 1.");
obj = pjp.proceed();
System.out.println("MoocAspect aroundInit 2.");
} catch (Throwable e) {
e.printStackTrace();
}
return obj;
}
Introductions
简介允许一个切面声明一个实现指定接口的通知对象 ,并且提供一个接口实现类来代表这些对象
由<aop:aspect>中的<aop:declare-parents>元素声明该元素用于声明所匹配的类型拥有一个新的parent(因此得名)
image.png
types-matching匹配的类型
implement-interface实现的接口
default-impl接口实现类
public interface Fit {
void filter();
}
public class FitImpl implements Fit {
@Override
public void filter() {
System.out.println("FitImpl filter.");
}
}
@Test
public void testFit() {
Fit fit = (Fit)super.getBean("aspectBiz");
fit.filter();
}为这个类aspectBiz的强制指定一个父类为FitImpl
所有配置式的aspect只支持单例
data:image/s3,"s3://crabby-images/e8c45/e8c45c5c02857703d410acddf72a9b78a3644b77" alt=""
data:image/s3,"s3://crabby-images/13c4b/13c4bc0e82d507dccff5c28c7fe5c2017243dc4a" alt=""
使用事务的时候经常用到
例子:
配置文件
<context:component-scan base-package="com.imooc.aop.schema"></context:component-scan>
<aop:config>
<aop:aspect id="concurrentOperationRetry" ref="concurrentOperationExecutor">
<aop:pointcut id="idempotentOperation"
expression="execution(* com.imooc.aop.schema.advisors.service.*.*(..)) " />
<!-- expression="execution(* com.imooc.aop.schema.service.*.*(..)) and -->
<!-- @annotation(com.imooc.aop.schema.Idempotent)" /> -->
<aop:around pointcut-ref="idempotentOperation" method="doConcurrentOperation" />
</aop:aspect>
</aop:config>
<bean id="concurrentOperationExecutor" class="com.imooc.aop.schema.advisors.ConcurrentOperationExecutor">
<property name="maxRetries" value="3" />
<property name="order" value="100" />
</bean>
类
public class ConcurrentOperationExecutor implements Ordered {
private static final int DEFAULT_MAX_RETRIES = 2;
private int maxRetries = DEFAULT_MAX_RETRIES;
private int order = 1;
public void setMaxRetries(int maxRetries) {
this.maxRetries = maxRetries;
}
public int getOrder() {
return this.order;
}
public void setOrder(int order) {
this.order = order;
}
public Object doConcurrentOperation(ProceedingJoinPoint pjp) throws Throwable {
int numAttempts = 0;
PessimisticLockingFailureException lockFailureException;
do {
numAttempts++;
System.out.println("Try times : " + numAttempts);
try {
return pjp.proceed();
} catch (PessimisticLockingFailureException ex) {
lockFailureException = ex;
}
} while (numAttempts <= this.maxRetries);
System.out.println("Try error : " + numAttempts);
throw lockFailureException;
}
}
Service类
@Service
public class InvokeService {
public void invoke() {
System.out.println("InvokeService ......");
}
public void invokeException() {
throw new PessimisticLockingFailureException("");
}
}
测试类:
@RunWith(BlockJUnit4ClassRunner.class)
public class TestAOPSchemaAdvisors extends UnitTestBase {
public TestAOPSchemaAdvisors() {
super("classpath:spring-aop-schema-advisors.xml");
}
@Test
public void testSave() {
InvokeService service = super.getBean("invokeService");
service.invoke();
System.out.println();
service.invokeException();
}
}
执行结果
data:image/s3,"s3://crabby-images/67d46/67d46332d6ee0233604ed52da11592fdc4583667" alt=""
第一次成功不再进行循环,第二次循环。
Spring Aop api
data:image/s3,"s3://crabby-images/35d93/35d9376d983d67b77b1cb3c402442a46d6d7c168" alt=""
data:image/s3,"s3://crabby-images/06a88/06a88ed1ad5711069c4169ef63515cc60c2dcfab" alt=""
data:image/s3,"s3://crabby-images/e1279/e12795475d6eb10c3fa4b1f48f6862662ab547ab" alt=""
public class MoocBeforeAdvice implements MethodBeforeAdvice {
@Override
public void before(Method method, Object[] args, Object target)
throws Throwable {
System.out.println("MoocBeforeAdvice : " + method.getName() + " " +
target.getClass().getName());
}
}
data:image/s3,"s3://crabby-images/9d4ac/9d4acd39717ee4e44a0c77b33a3a2d16f1198ede" alt=""
data:image/s3,"s3://crabby-images/bfc25/bfc252f9ea522dbb76b7f0e333e61cb0623b8cad" alt=""
public class MoocThrowsAdvice implements ThrowsAdvice {
public void afterThrowing(Exception ex) throws Throwable {
System.out.println("MoocThrowsAdvice afterThrowing 1");
}
public void afterThrowing(Method method, Object[] args, Object target, Exception ex) throws Throwable {
System.out.println("MoocThrowsAdvice afterThrowing 2 : " + method.getName() + " " +
target.getClass().getName());
}
data:image/s3,"s3://crabby-images/4f282/4f282bae8ec4eedd3deabcc74346523a7a7f610a" alt=""
public class MoocAfterReturningAdvice implements AfterReturningAdvice {
@Override
public void afterReturning(Object returnValue, Method method,
Object[] args, Object target) throws Throwable {
System.out.println("MoocAfterReturningAdvice : " + method.getName() + " " +
target.getClass().getName() + " " + returnValue);
}
}
data:image/s3,"s3://crabby-images/4af34/4af34e3378f501a5d715cb7d70d85c6872ad4493" alt=""
public class MoocMethodInterceptor implements MethodInterceptor {
@Override
public Object invoke(MethodInvocation invocation) throws Throwable {
System.out.println("MoocMethodInterceptor 1 : " + invocation.getMethod().getName() + " " +
invocation.getStaticPart().getClass().getName());
Object obj = invocation.proceed();
System.out.println("MoocMethodInterceptor 2 : " + obj);
return obj;
}
}
data:image/s3,"s3://crabby-images/bd7f1/bd7f139f59e419c6d5b0ed65bb4546cb65b49d6f" alt=""
data:image/s3,"s3://crabby-images/05371/05371a2658079c5bd452d9795a45123c19441f40" alt=""
data:image/s3,"s3://crabby-images/74f6b/74f6bac6f6e7f1ffd84ca3db4881e0c51c370401" alt=""
data:image/s3,"s3://crabby-images/ccdaa/ccdaaa4dfce08ce51cc83e09841e61aa3f4c4297" alt=""
public interface Lockable {
void lock();
void unlock();
boolean locked();
}
public class LockMixin extends DelegatingIntroductionInterceptor implements Lockable {
private static final long serialVersionUID = 6943163819932660450L;
private boolean locked;
public void lock() {
this.locked = true;
}
public void unlock() {
this.locked = false;
}
public boolean locked() {
return this.locked;
}
public Object invoke(MethodInvocation invocation) throws Throwable {
if (locked() && invocation.getMethod().getName().indexOf("set") == 0) {
throw new RuntimeException();
}
return super.invoke(invocation);
}
}
public class LockMixinAdvisor extends DefaultIntroductionAdvisor {
private static final long serialVersionUID = -171332350782163120L;
public LockMixinAdvisor() {
super(new LockMixin(), Lockable.class);
}
}
data:image/s3,"s3://crabby-images/fe44c/fe44c8c7f01430eee012b652bb037d23a85dc35e" alt=""
springAOP代理的核心类
data:image/s3,"s3://crabby-images/1dc27/1dc273632c28ce569eb4ad6d2e27483bf08aae63" alt=""
foo引用ProxyFactoryBean,得到的对象是getOpject()方法创建的
data:image/s3,"s3://crabby-images/c8509/c8509eece39096afc0cb00b8b9da6ed78f92caa6" alt=""
data:image/s3,"s3://crabby-images/89c3c/89c3c2a1180fbc2573084e548bb1dfe3b207e6fb" alt=""
data:image/s3,"s3://crabby-images/2db4f/2db4fae73dc7582a582c09393d9749829e60c31f" alt=""
创建一个基于接口的代理,getOpject()方法返回的就是target
image.png
配置:
<?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:context="http://www.springframework.org/schema/context"
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/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd">
<bean id="moocBeforeAdvice" class="com.imooc.aop.api.MoocBeforeAdvice"></bean>
<bean id="moocAfterReturningAdvice" class="com.imooc.aop.api.MoocAfterReturningAdvice"></bean>
<bean id="moocMethodInterceptor" class="com.imooc.aop.api.MoocMethodInterceptor"></bean>
<bean id="moocThrowsAdvice" class="com.imooc.aop.api.MoocThrowsAdvice"></bean>
<!-- <bean id="bizLogicImplTarget" class="com.imooc.aop.api.BizLogicImpl"></bean> -->
<!-- <bean id="pointcutBean" class="org.springframework.aop.support.NameMatchMethodPointcut"> -->
<!-- <property name="mappedNames"> -->
<!-- <list> -->
<!-- <value>sa*</value> -->
<!-- </list> -->
<!-- </property> -->
<!-- </bean> -->
<!-- <bean id="defaultAdvisor" class="org.springframework.aop.support.DefaultPointcutAdvisor"> -->
<!-- <property name="advice" ref="moocBeforeAdvice" /> -->
<!-- <property name="pointcut" ref="pointcutBean" /> -->
<!-- </bean> -->
<!-- <bean id="bizLogicImpl" class="org.springframework.aop.framework.ProxyFactoryBean"> -->
<!-- <property name="target"> -->
<!-- <ref bean="bizLogicImplTarget"/> -->
<!-- </property> -->
<!-- <property name="interceptorNames"> -->
<!-- <list> -->
<!-- <value>defaultAdvisor</value> -->
<!-- <value>moocAfterReturningAdvice</value> -->
<!-- <value>moocMethodInterceptor</value> -->
<!-- <value>moocThrowsAdvice</value> -->
<!-- </list> -->
<!-- </property> -->
<!-- </bean> -->
<!-- <bean id="bizLogicImplTarget" class="com.imooc.aop.api.BizLogicImpl"></bean> -->
<!-- <bean id="bizLogicImpl" class="org.springframework.aop.framework.ProxyFactoryBean"> -->
<!-- <property name="proxyInterfaces"> -->
<!-- <value>com.imooc.aop.api.BizLogic</value> -->
<!-- </property> -->
<!-- <property name="target"> -->
<!-- <bean class="com.imooc.aop.api.BizLogicImpl"></bean> -->
<!-- <ref bean="bizLogicImplTarget"/> -->
<!-- </property> -->
<!-- <property name="interceptorNames"> -->
<!-- <list> -->
<!-- <value>moocBeforeAdvice</value> -->
<!-- <value>moocAfterReturningAdvice</value> -->
<!-- <value>moocMethodInterceptor</value> -->
<!-- <value>moocThrowsAdvice</value> -->
<!-- <value>mooc*</value> -->
<!-- </list> -->
<!-- </property> -->
<!-- </bean> -->
<bean id="baseProxyBean" class="org.springframework.aop.framework.ProxyFactoryBean"
lazy-init="true" abstract="true"></bean>
<bean id="bizLogicImpl" parent="baseProxyBean">
<property name="target">
<bean class="com.imooc.aop.api.BizLogicImpl"></bean>
</property>
<property name="proxyInterfaces">
<value>com.imooc.aop.api.BizLogic</value>
</property>
<property name="interceptorNames">
<list>
<value>moocBeforeAdvice</value>
<value>moocAfterReturningAdvice</value>
<value>moocMethodInterceptor</value>
<value>moocThrowsAdvice</value>
</list>
</property>
</bean>
</beans>
实现bean:
public class BizLogicImpl implements BizLogic {
public String save() {
System.out.println("BizLogicImpl : BizLogicImpl save.");
return "BizLogicImpl save.";
// throw new RuntimeException();
}
}
接口:
public interface BizLogic {
String save();
}
测试类:
@RunWith(BlockJUnit4ClassRunner.class)
public class TestAOPAPI extends UnitTestBase {
public TestAOPAPI() {
super("classpath:spring-aop-api.xml");
}
@Test
public void testSave() {
BizLogic logic = (BizLogic)super.getBean("bizLogicImpl");
logic.save();
}
}
打印结果:
data:image/s3,"s3://crabby-images/d27ba/d27ba1f87e8e31897aaecaac183b9d8111b7440f" alt=""
data:image/s3,"s3://crabby-images/c55e7/c55e7e4de5c6a6014269fd26dbd8b3b4e137cd17" alt=""
data:image/s3,"s3://crabby-images/889c9/889c937bda4ae1c68a3476bc7a2d40ab7c220233" alt=""
data:image/s3,"s3://crabby-images/990cb/990cba037a52664a7b052674d52342e6a722534d" alt=""
data:image/s3,"s3://crabby-images/28c74/28c741e116ff9951391456dec1e1ea1a17462f81" alt=""
实现Interceptor的拦截器。
data:image/s3,"s3://crabby-images/2c143/2c1438da4a8e44c0176c06d9a9bc5d3611965746" alt=""
data:image/s3,"s3://crabby-images/73869/7386967ac1389acc78805e3b5a4fc73ddefaece3" alt=""
data:image/s3,"s3://crabby-images/124c2/124c2a34028841fd24327613672679b391155ce7" alt=""
data:image/s3,"s3://crabby-images/bf355/bf35544fa2a244d23f007914b1a9c0edf1e47559" alt=""
data:image/s3,"s3://crabby-images/8cb94/8cb942a84dd93a9b86c1d9dfd203556a364e6074" alt=""
Aspectj
data:image/s3,"s3://crabby-images/e2eb7/e2eb753ba20a2ef0ef837e7892e9bb76bbdd77db" alt=""
Aspectj的两种方式
data:image/s3,"s3://crabby-images/da135/da1353b82fa7f2737746c9f66ca23d19fa6deb5b" alt=""
data:image/s3,"s3://crabby-images/b85cc/b85cc44822e6cf971dda8f224ae489a3601d5f03" alt=""
data:image/s3,"s3://crabby-images/04885/04885e39a54660df1f1e03f8d73ab558055c98dd" alt=""
data:image/s3,"s3://crabby-images/390d2/390d271a6792b4fe14e7910756a0c88577c2c2e5" alt=""
切入点定义方式
data:image/s3,"s3://crabby-images/204d5/204d50ffa349821266adea15b56f7d4d6b5d85f8" alt=""
public class MoocAspect {
@Pointcut("execution(* com.imooc.aop.aspectj.biz.*Biz.*(..))")
//biz结尾的方法
public void pointcut() {}
//包下的任何类
@Pointcut("within(com.imooc.aop.aspectj.biz.*)")
public void bizPointcut() {}
}
data:image/s3,"s3://crabby-images/7365c/7365c1886b93cc624688e6cb8ffe9f78db055c6e" alt=""
data:image/s3,"s3://crabby-images/be705/be7059014abf9f90a86837962a84f606840e7a33" alt=""
data:image/s3,"s3://crabby-images/53713/5371348f874331d8a1578ee13b6000f1866a761e" alt=""
data:image/s3,"s3://crabby-images/78375/783759a1f3f259ae4c87d55a2e4f3d5ddfaadabe" alt=""
data:image/s3,"s3://crabby-images/a5110/a5110421808fa47730388ddb77db2717deb261ac" alt=""
data:image/s3,"s3://crabby-images/2fcd0/2fcd0910ccf58f831d482425f43d6843d7f27194" alt=""
data:image/s3,"s3://crabby-images/f0eff/f0eff646045087b9923ab656d164413aaa5a43c0" alt=""
data:image/s3,"s3://crabby-images/05a0e/05a0e7a024064087534084806e328f48779ce462" alt=""
@Component
@Aspect
public class MoocAspect {
@Pointcut("execution(* com.imooc.aop.aspectj.biz.*Biz.*(..))")
public void pointcut() {}
@Pointcut("within(com.imooc.aop.aspectj.biz.*)")
public void bizPointcut() {}
@Before("pointcut()")
public void before() {
System.out.println("Before.");
}
@Before("pointcut() && args(arg)")
public void beforeWithParam(String arg) {
System.out.println("BeforeWithParam." + arg);
}
@Before("pointcut() && @annotation(moocMethod)")
public void beforeWithAnnotaion(MoocMethod moocMethod) {
System.out.println("BeforeWithAnnotation." + moocMethod.value());
}
@AfterReturning(pointcut="bizPointcut()", returning="returnValue")
public void afterReturning(Object returnValue) {
System.out.println("AfterReturning : " + returnValue);
}
@AfterThrowing(pointcut="pointcut()", throwing="e")
public void afterThrowing(RuntimeException e) {
System.out.println("AfterThrowing : " + e.getMessage());
}
@After("pointcut()")
public void after() {
System.out.println("After.");
}
@Around("pointcut()")
public Object around(ProceedingJoinPoint pjp) throws Throwable {
System.out.println("Around 1.");
Object obj = pjp.proceed();
System.out.println("Around 2.");
System.out.println("Around : " + obj);
return obj;
}
}
@Service
public class MoocBiz {
@MoocMethod("MoocBiz save with MoocMethod.")
public String save(String arg) {
System.out.println("MoocBiz save : " + arg);
// throw new RuntimeException(" Save failed!");
return " Save success!";
}
}
data:image/s3,"s3://crabby-images/eccc5/eccc5c7e33890cf075ccce4df2ba892a556d1758" alt=""
data:image/s3,"s3://crabby-images/7dd5d/7dd5d7ab63af68599e810ab808fe5b8f30ab1ad5" alt=""
网友评论