欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Spring 核心之AOP怎么使用(入门介绍)

程序员文章站 2022-06-21 14:26:30
Spring AOP一、spring(春天)核心之 aop(重点)二、aop怎么玩?三、动态代理(重点)黑客入侵四、aop的术语一、spring(春天)核心之 aop(重点)aop:Aspect Oriented Programming 面向切面编程 ,是一种思想。oop: Object Orinted programming 面向对象编程 ,是一种思想。AOP可以说是OOP的补充和完善。OOP引入封装、继承、多态等概念来建立一种对象层次结构,在类的层次上进行抽取。不过OOP允许开发...



一、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

相关标签: spring java aop