AOP核心概念
1、横切关注点
对哪些方法进行拦截,拦截后怎么处理,这些关注点称之为横切关注点
2、切面(aspect)
类是对物体特征的抽象,切面就是对横切关注点的抽象
3、连接点(joinpoint)
被拦截到的点,因为spring只支持方法类型的连接点,所以在Spring中连接点指的就是被拦截到的方法,实际上连接点还可以是字段或者构造器
4、切入点(pointcut)
对连接点进行拦截的定义
5、通知(advice)
所谓通知指的就是指拦截到连接点之后要执行的代码,通知分为前置、后置、异常、最终、环绕通知五类
6、目标对象
代理的目标对象
7、织入(weave)
8、引入(introduction)
在不修改代码的前提下,引入可以在运行期为类动态地添加一些方法或字段
Spring 实现AOP所需要的包:
1、Spring提供的jar包
2、aopalliance.jar
3、aspectjweaver.jar
Spring 实现AOP的方式:
1、Java动态代理
该方法针对接口的实例创建代理
applicationContext.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" xmlns:tx="http://www.springframework.org/schema/tx" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.2.xsd"> <bean id="concreteImplementor" class="com.marving.aop.ConcreteImplementor" /> <bean id="interceptorHandler" class="com.marving.aop.InterceptorHandler" /> <aop:config> <aop:aspect id="interceptor" ref="interceptorHandler"> <aop:pointcut id="addAllMethod" expression="execution(* com.marving.aop.Abstration.*(..))" /> <aop:before method="doSomething" pointcut-ref="addAllMethod" /> <aop:after method="doSomething" pointcut-ref="addAllMethod" /> </aop:aspect> </aop:config> </beans>
其中Abstration为接口,ConcreteImplementor为实现类,InterceptorHandler为代理拦截类。
public interface <span style="font-size:12px;">Abstration</span> { public void operation() }
//具体实现化角色 public class ConcreteImplementor implements Implementor{ @Override public void operation() { System.out.println("ConcreteImplementor"); } }
public class InterceptorHandler{ public void printTime(){ System.out.println("CurrentTime = " + System.currentTimeMillis()); } }
2、CGLIB生成代理
CGLIB针对代理对象为类的情况使用。
通过实现MethodInterceptor接口,并实现 public Object intercept(Object obj, Method m, Object[] args,MethodProxy proxy) throws Throwable方法生成代理。
3、BeanNameAutoProxyCreator实现AOP
Spring为我们提供了自动代理机制,让容器为我们自动生成代理,把我们从烦琐的配置工作中解放出来,在内部,Spring 使用BeanPostProcessor自动地完成这项工作。
具体配置如下:
<bean id="MyInterceptor" class="com.yesjpt.interceptor. MyInterceptor"></bean> <bean class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator"> <property name="beanNames"> <list> <value>*Service</value> </list> </property> <property name="interceptorNames"> <list> <value>MyInterceptor</value> </list> </property> </bean>
其中*Service 为需要拦截代理的bean,以Service结尾的都 被拦截,并使用MyInterceptor 进行拦截,可配置多个拦截器,按顺序执行。
import java.lang.reflect.Method; import org.aopalliance.intercept.MethodInterceptor; import org.aopalliance.intercept.MethodInvocation; /** * @author * */ public class MyInterceptor implements MethodInterceptor{ @Override public Object invoke(MethodInvocation invocation) throws Throwable { Method method = invocation.getMethod();//获取被拦截的方法 Object[] arguments = invocation.getArguments();//获取拦截方法的参数 /* * 特殊,某些权限需要做特殊处理 * 比如用户信息权限,在方法执行完毕返回的时候,要将电话号码与邮箱抹除 */ //环绕通知前置特殊处理 this.beforeReslove(); Object proceed = invocation.proceed();//调用目标方法 //环绕通知后置特殊处理 proceed = this.afterReslove(); return proceed; } private Object afterReslove() { System.out.println("CurrentTime = " + System.currentTimeMillis()); return null; } private void beforeReslove() { System.out.println("CurrentTime = " + System.currentTimeMillis()); } }
4、使用注解AspectJ实现AOP
ApplicationContext.xml 加入
<aop:aspectj-autoproxy/>
创建切面处理类
package com.marving.aop; import java.util.Arrays; import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Pointcut; import org.springframework.stereotype.Component; @Aspect @Component public class AspectHandler { @Pointcut("execution(* com.marving.service.BaseServ+.*(..))") private void doMethod() { } /** * This is the method which I would like to execute before a selected method * execution. */ @Before("doMethod()") public void beforeAdvice() { System.out.println("before method invoked."); } /** * This is the method which I would like to execute after a selected method * execution. */ @After("doMethod()") public void afterAdvice() { System.out.println("after method invoked."); } // 配置controller环绕通知,使用在方法aspect()上注册的切入点 @Around("doMethod()") public Object around(ProceedingJoinPoint pjp) throws Throwable{ Object result = null; String methodName = pjp.getSignature().getName(); try { System.out.println("The method [" + methodName + "] begins with " + Arrays.asList(pjp.getArgs())); result = pjp.proceed(); } catch (Throwable e) { System.out.println("The method [" + methodName + "] occurs expection : " + e); throw new RuntimeException(e); } System.out.println("The method [" + methodName + "] ends"); return result; } }
通过表达式execution(* com.marving.service.BaseServ+.*(..)) 匹配切入点函数,并使用@Before@After@Around 对所拦截方法执行前、中、后进行拦截并执行处理函数。
@Around @Before @After三个注解的区别@Before是在所拦截方法执行之前执行一段逻辑。@After 是在所拦截方法执行之后执行一段逻辑。@Around是可以同时在所拦截方法的前后执行一段逻辑。
值得注意的是,Around在拦截方法后,需要返回一个方法执行结果,否则,原方法不能正常执行。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持小牛知识库。
本文向大家介绍C#实现单例模式的几种方法总结,包括了C#实现单例模式的几种方法总结的使用技巧和注意事项,需要的朋友参考一下 介绍 单例模式是软件工程学中最富盛名的设计模式之一。从本质上看,单例模式只允许被其自身实例化一次,且向外部提供了一个访问该实例的接口。通常来说,单例对象进行实例化时一般不带参数,因为如果不同的实例化请求传递的参数不同的话会导致问题的产生。(若多个请求都是传递的同样的参数的话,
语言是一种规范,要想使用它,需要将其实体化,用编译原理的话讲,可以大致分为编译器和解释器,而根据之前说的,如果把目标代码的解释过程看做解析过程,比如x86机器指令load到cpu时,根据不同的指令进行不同的运算,以及字节码的分派等,这个界限是模糊的 因此,这个事情可以这么讲,假设我们已经有一台机器M,它可以接收A语言并完成计算工作,则对于高级语言B来说,我们只要将其转换成A语言即可。根据实际情况,
本文向大家介绍java 多线程的几种实现方法总结,包括了java 多线程的几种实现方法总结的使用技巧和注意事项,需要的朋友参考一下 java 多线程的几种实现方法总结 1.多线程有几种实现方法?同步有几种实现方法? 多线程有两种实现方法,分别是继承Thread类与实现Runnable接口 同步的实现方面有两种,分别是synchronized,wait与notify wait():使一个线程处于等待
本文向大家介绍Java单例模式实现的几种方式,包括了Java单例模式实现的几种方式的使用技巧和注意事项,需要的朋友参考一下 Java单例模式实现的几种方式 单例模式好多书上都是这么写的: 但是实际开发中是不会这么写的,因为有一个严重的问题,多线程并发访问的时候,可能会产生多个实例!! 下面列举几个常用的方法: 1.使用synchronized 关键字 2.加锁 3.利用静态变量: 以上就
本文向大家介绍Android实现View滑动的几种方式,包括了Android实现View滑动的几种方式的使用技巧和注意事项,需要的朋友参考一下 什么是View?实现View滑动的方式有哪些? 1. 关于View我们需要知道的 (1)什么是View? Android中的View类是所有UI控件的基类(Base class),也就是说我们平时所有到的各种UI控件,比如Button、ImagView等等
本文向大家介绍JS中继承实现的几种方式?相关面试题,主要包含被问及JS中继承实现的几种方式?时的应答技巧和注意事项,需要的朋友参考一下 参考回答: 1、原型链继承,将父类的实例作为子类的原型,他的特点是实例是子类的实例也是父类的实例,父类新增的原型方法/属性,子类都能够访问,并且原型链继承简单易于实现,缺点是来自原型对象的所有属性被所有实例共享,无法实现多继承,无法向父类构造函数传参。 2、构造继