本文主要是介绍AOP中xlm配置和注解配置,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
概念:Aspect Oriented Programming 面向切面编程
面向切面编程时一种改编成思想。就是通过预编译和运行期的动态代理,实现程序功能的统一维护。spring是对这个思想的实现。
动态代理:简而言之就是在不修改源代码的情况下,对其中的方法进行增强,实现调用者和实现者的解耦。常用的动态代理有:
- JDK代理:基于接口的动态代理技术
- cglib代理:基于父类的动态代理技术
对一些概念理解:
- Target(目标对象)要增强的对象
- Proxy(代理)增强后的对象。
- Joinpoint(连接点)要增强的方法
- Pointcut(切点)被增强的方法
- Advice(通知)增强方法
- Aspect(切面)通知+切点。就是增强后的方法。
- Weaving(织入)通知和切点相结合的过程。创建增强的代理对象的过程。
基于JDK
这个代理基于Java的反射机制实现的。主要涉及到java.lang.reflect包中的Proxy和InvocationHandler。
底层代码简单写了一下
public class Proxy动态增强 {public static void main(String[] args) {/** jdk的动态增强,基于接口。*///创建要增强的对象final Target target = new Target ();//创建增强类,存的是增强的功能final Aspect aspect = new Aspect();//使用动态代理将要增强的对象和功能组织到一块儿Egg ss = (Egg) Proxy.newProxyInstance(target.getClass().getClassLoader(),//目标对象的类加载器target.getClass().getInterfaces(),//目标对象相同接口字节码对象数组new InvocationHandler(){//调用代理对象的方法,实际执行的都是invoke方法。public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {//调用增强功能aspect.before();//执行目标原有方法method.invoke(target ,args);//调用增强功能aspect.after();return null;}});ss.fight();}
}
浅谈一下为什么用final修饰
在jdk1.8以后就不用了,之前需要。
main方法执行开始,进栈内存,成员变量进栈内存,常量进常量池。new对象的时候,对象在对内存。栈内存存储的是new出来的对象的地址值。在堆内存的对象引用被赋值的成员变量时,是不可以的。就相当于从堆内存找栈内存的数据。而在方法执行完毕后,成员变量随着方法的结束而消失,到那时堆内存的对象暂时不会消失。所以强大编码软件就会报错。被final修饰的变量为常量,存在于常量池中。常量池中的数据随着程序的结束而消失,所以堆内存的对象在寻找它需要的变量时,可以直接从常量池中寻找。所以用final修饰。
1.8以后,jdk在编译.class文件的时候就会加上final。所以不用。
1 基于xml
-
导入坐标spring-context和context依赖的aop
<dependency><groupId>org.springframework</groupId><artifactId>spring-context</artifactId><version>5.0.5.RELEASE</version></dependency><!-- aspectj的织入 --><dependency><groupId>org.aspectj</groupId><artifactId>aspectjweaver</artifactId><version>1.8.13</version></dependency>
-
导入aop的命名空间
<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/contexthttp://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/aophttp://www.springframework.org/schema/aop/spring-aop.xsdhttp://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsd">
-
在配置文件中配置织入关系
<!--目标对象--><bean id="target" class="cn.itcast.jdkProxyXml.Target"/><!--切面对象--><bean id="myAspect" class="cn.itcast.jdkProxyXml.MyAspect"/><!--配置织入:告诉spring哪些方法(切点)需要哪些增强(前置、后置... ... )--><aop:config><!--声明切面类--><aop:aspect ref="myAspect"><aop:pointcut id="myPoint" expression="execution(public void cn.itcast.jdkProxyXml.Target.save())"/><!--生成切面:切点+通知--><aop:after method="after" pointcut-ref="myPoint"/><aop:around method="around" pointcut-ref="myPoint"/><aop:before method="before" pointcut-ref="myPoint"/></aop:aspect></aop:config> </beans>
黄色箭头为提取切点表达式。
蓝色为通知类型
红色为切点表达式的写法
浅谈为什么用aspectjweaver这个jar包
spring中已经有aop的jar包了,为什么还要引入aspectjweaver这个。简单的说就是后面这个比spring自己坐的jar包要好。
如果仅仅用xlm的配置方法,原生的aop就可以了,但是要使用注解,就的引入aspectjweaver这个jar包。
2 基于注解
在配置文件中配置组键扫描和自动代理。
<!--组件扫描-->
<context:component-scan base-package="com.itheima.aop"/><!--aop的自动代理-->
<aop:aspectj-autoproxy></aop:aspectj-autoproxy>
注解配置与xml的大同小异
切点表达式的抽取
@@Component("myAspect")
@Aspect
public class MyAspect {@Before("MyAspect.myPoint()")public void before(){System.out.println("前置代码增强.....");}@Pointcut("execution(* com.itheima.aop.*.*(..))")public void myPoint(){}
}
声明切面类
基于cglib
这篇关于AOP中xlm配置和注解配置的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!