Spring 核心之AOP怎么使用(入门介绍)
Spring AOP
一、spring(春天)核心之 aop(重点)
aop:Aspect Oriented Programming 面向切面编程 ,是一种思想。
oop: Object Orinted programming 面向对象编程 ,是一种思想。
AOP可以说是OOP的补充和完善。OOP引入封装、继承、多态等概念来建立一种对象层次结构,在类的层次上进行抽取。不过OOP允许开发者定义纵向的关系,但并不适合定义横向的关系,例如日志功能。日志代码往往横向地散布在所有对象层次中,而与它对应的对象的核心功能毫无关系对于其他类型的代码,如安全性、异常处理和透明的持续性也都是如此,这种散布在各处的无关的代码被称为横切(cross cutting),在OOP设计中,它导致了大量代码的重复,而不利于各个模块的重用。
AOP技术恰恰相反,它利用一种称为"横切"的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其命名为"Aspect",即切面。所谓"切面",简单说就是那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块之间的耦合度,并有利于未来的可操作性和可维护性。
总结:
oop:纵向抽取
好处:
1)提高可重用性
2)提高可扩展和可维护
3)多态
aop:横向向抽取
好处:
1)提高可重用性
2)提高可扩展和可维护
二、aop怎么玩?
1.aop底层原理:动态代理技术
2. jdk动态代理和cglib动态代理
代理========>经纪人、中介
三、动态代理(重点)
黑客入侵
1)jdk黑客: 只能入侵实现接口的对象。 针对接口实现类
jdk接口:InvocationHandler
==>JDK动态代理技术
## jdk动态代理(只能动态代理有接口的类)
1、先创建一个接口
public interface UserDao { int add(int a, int b); }
2、实现接口
public class UserDaoImpl implements UserDao { @Override public int add(int a, int b) { System.out.println("调用UserDaoImpl的add方法"); return a+b; } }
3、创建JdkHk实现InvocationHandler接口
public class JdkHk implements InvocationHandler{ //目标对象 private Object target; public JdkHk() { } //有参构造,传入参数 public JdkHk(UserDao userDao) { this.target=userDao; } /**
* 黑客类入侵的方法
* proxy:代理对象
* method:入侵的目标的方法
* args:入侵的目标的方法的参数
*/ @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("1、鉴权成功"); //干坏事 args[0]=11; args[1]=22; //调用目标方法 Object result = method.invoke(target, args); System.out.println("2、保存处理日志成功"); return result; } public Object getTarget() { return target; } public void setTarget(Object target) { this.target = target; } }
4、测试类
public class Test01 { public static void main(String[] args) { //客户端---调用目标类(UserDaoImpl)的目标方法add方法 // UserDao userDao = new UserDaoImpl(); // int result = userDao.add(3, 5); // System.out.println("result==>:"+result); //目标对象 UserDao targert = new UserDaoImpl(); //黑客对象 JdkHk jdkHk = new JdkHk(targert); //代理对象 /*
* loader 类加载器
* ClassLoader.getSystemClassLoader() 获取当前程序的类加载器
* interfaces 目标的实现的接口class
* h invocationHandler对象--黑客对象
* 字节码拼接技术
*/ UserDao userDao = (UserDao) Proxy.newProxyInstance(ClassLoader.getSystemClassLoader(), new Class[] {UserDao.class}, jdkHk); //客户端 int result = userDao.add(45, 5); System.out.println("result==>"+result); } }
2)spring黑客:针对类(aspect包—spring提供的)
aopalliance: MethodInteceptor 当类实现接口内部用的jdk黑客 如果类没有实现接口 使用cglib动态代理
==>可以使用JDK动态代理(实现接口类) 也可以使用Cglib动态代理(类或实现接口类)
1、创建一个不实现接口的类
public class UserDaoImpl2 { public int add(int a, int b) { System.out.println("调用UserDaoImpl的add方法"); return a+b; } }
2、创建springHk类实现MethodInteceptor 接口
public class SpringHk implements MethodInterceptor{ /*
* invocation : 目标对象的目标方法
*/ @Override public Object invoke(MethodInvocation invocation) throws Throwable { System.out.println("1.鉴权"); //调用目标方法 Object result = invocation.proceed(); System.out.println("1.日志留痕"); return result; } }
3、测试类(这里通过依赖注入的方式来创建对象,需要配置文件spring-beans.xml)
@RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration("classpath:spring-beans.xml") public class SpringHkTest { @Autowired//默认按类型注入,通过@Qualifier修改为按名称注入 @Qualifier("userDaoProxy") private UserDaoImpl2 userDao;//cglib @Autowired//默认按类型注入,通过@Qualifier修改为按名称注入 @Qualifier("userDaoProxy2") private UserDaoImpl userDao2;//cglib @Test public void test01() { //目标没有实现接口 System.out.println(userDao.add(3, 5)); System.out.println(userDao2.add(3, 5)); } }
spring-beans.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:p="http://www.springframework.org/schema/p" xsi:schemaLocation="http://www.springframework.org/schema/beans
https://www.springframework.org/schema/beans/spring-beans.xsd"> <!--1.目标对象 --> <bean id="userDao" class="com.cc.dao.impl.UserDaoImpl2"></bean> <bean id="userDaoInter" class="com.cc.dao.impl.UserDaoImpl"></bean> <!--2.spring 黑客对象 --> <bean id="springHk" class="com.cc.proxy.SpringHk"> </bean> <bean id="jdkHk" class="com.cc.proxy.JdkHk"> <!-- <constructor-arg name="target" ref="userDaoInter"></constructor-arg> --> </bean> <!--3.代理对象 :Cglib动态代理 --> <bean id="userDaoProxy" class="org.springframework.aop.framework.ProxyFactoryBean"> <!-- 3.1 目标对象 --> <property name="target" ref="userDao"></property> <!-- 3.2 spring 黑客对象 --> <property name="interceptorNames"> <array> <value>springHk</value> </array> </property> </bean> <!-- JDK动态代理 --> <bean id="userDaoProxy2" class="org.springframework.aop.framework.ProxyFactoryBean"> <!-- 3.2 目标对象 --> <property name="proxyInterfaces" value="com.cc.dao.UserDao"></property> <!-- 3.1 目标对象 --> <property name="target" ref="userDaoInter"></property> <!-- 3.2 spring 黑客对象 --> <property name="interceptorNames"> <array> <value>springHk</value> </array> </property> </bean> </beans>
四、aop的术语
1、横切关注点
对哪些方法进行拦截,拦截后怎么处理,这些关注点称之为横切关注点
2、切面(aspect)
类是对物体特征的抽象,切面就是对横切关注点的抽象
3、连接点(joinpoint)
被拦截到的点,因为Spring只支持方法类型的连接点,所以在Spring中连接点指的就是被拦截到的方法,实际上连接点还可以是字段或者构造器
4、切入点(pointcut)
对连接点进行拦截的定义
5、通知(advice)
所谓通知指的就是指拦截到连接点之后要执行的代码,通知分为前置、后置、异常、最终、环绕通知五类
6、目标对象
代理的目标对象
7、植入(weave)
将切面应用到目标对象并导致代理对象创建的过程
8、引入(introduction)
在不修改代码的前提下,引入可以在运行期为类动态地添加一些方法或字段
aop:将对象的目标方法通过动态代理技术进行植入增强,从而达到
增强方法的目的
本文地址:https://blog.csdn.net/GanYangBa/article/details/108859833