【Spring】一文带你彻底搞懂IOC、AOP

目录

首先简单了解一下什么是spring框架

什么是IOC?

什么是依赖注入(DI)?

控制反转和依赖注入又有什么关系?

AOP是什么?

SpringAOP的实现

说了这么多抽象概念,举个实例方便理解


首先简单了解一下什么是spring框架

Spring是一个轻量级的框架,通过IOC达到松耦合的目的,使用AOP可以分离应用业务逻辑和系统服务进行内聚性的开发(不改变原有逻辑的同时对业务进行增强),不够配置各种组件的时候相对繁琐,因此后面演进除了SpringBoot框架。

使用Spring之后可以让我们不用过多关注底层的技术细节,而是更加关注业务上的逻辑的实现

什么是IOC?

Inverse of Control ------控制反转,是一种思想,这种控制反转的思想主要指的是将对象的创建、组装、管理都从代码中自己实现转移到了外部容器中来帮我们进行实现。在传统的开发方式当中,我们直接手写代码去主动创建和组装对象(将对象所需要的属性注入);在IOC思想中,这个过程被反转了,即由外部容器负责创建和管理对象。

在IOC中,我们将应用程序设计成一个个的组件,每个组件提供一定的功能,并通过接口与其他组件进行交互。通过IOC容器,我们可以把这些组件注册并配置,容器负责根据配置信息创建组件实例,并维护它们之间的依赖关系和生命周期。

什么是依赖注入(DI)?

依赖注入指的是将对象所依赖的其他对象(即依赖)注入到当前对象之中,而不是由对象自己去创建或查找他要依赖的对象

在传统的开发方式当中,对象通过直接创建或查找依赖的对象来获取所需的依赖。

而在DI中,对象不负责创建或查找的过程,而是通过构造函数、工厂方法或属性的方式接收依赖对象。这样,依赖对象的创建和管理由外部容器负责,对象只需要专注于自身的功能实现。

依赖注入可以通过构造函数注入、setter方法注入或接口注入等方式来实现。

控制反转和依赖注入又有什么关系?

在Spring中,IOC是一种设计思想,而DI是IOC的一种具体的实现方式。实际上DI是IOC的实现技术中的一种,它通过依赖注入来实现控制反转。

IOC思想提倡将对象的设计与创建交给外部容器,并通过xml配置文件、注解、或者Java Config等方式来描述和配置对象之间的依赖关系,

Spring框架提供了一个IOC容器,即ApplicationContext,它就实现类DI的功能。在Spring中,我们可以使用XML配置文件、注解、或Java Config等方式来描述和配置对象之间的依赖关系,然后Spring容器会根据配置信息来创建对象并完成注入。

AOP是什么?

AOP即面向切面编程,可以将那些与业务不想关但是很多业务都需要调用的代码提取出来,思想就是不侵入原有代码的同时对功能进行增强。

AOP通过定义一个切面,切面可以横切到应用程序的多个模块中,并添加增强的行为。这样我们就可以将通用的功能逻辑从业务逻辑中解耦出来,提高代码的可维护性和重用性。

切面由切点(Pointcut)和通知(Advice组成)

  1. 切点定义的是在程序中的哪个位置进行拦截,通常使用表达式来指定匹配的连接点(join Point)
  2. 通知定义了要在切点执行的代码,可以在切点之前、切点之后或者周围执行特定的逻辑
  3. 常见的通知类包括前置通知(Before Advice)、后置通知(After Advice)、异常通知(After Throwing Advice)和环绕通知(Around Advice)等。

SpringAOP的实现

SpringAOP是基于动态代理实现的,动态代理有两种,一种是JDK动态代理,另一种是Cglib动态代理

jdk动态代理是利用反射的原理来实现的,需要调用反射包下的Proxy类的newProxyInstance方法来返回代理对象,这个方法中有三个参数,分别是用于加载代理类的类加载器被代理类实现的接口的class数组用于增强方法的InvocatioHandler实现类

cglib动态代理原理是利用asm开源包来实现的,是把被代理类的class文件加载进来,通过修改它的字节码生成子类来处理

jdk动态代理要求代理类必须有实现的接口,生成的动态代理类会和代理类实现同样的接口,cglib则,生成的动态代理类会继承被代理类。Spring默认使用jdk动态代理,当要被代理的类没有实现任何接口的时候采用cglib

说了这么多抽象概念,举个实例方便理解

这是一个实现统计controller路由访问次数的代码

java 复制代码
package com.qcby.springbootdemo.aop;
 
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.Signature;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;
 
import java.util.HashMap;
import java.util.Map;
 
@Aspect
@Component
public class MethodCount {
 
    // 声明一个Map类型的对象,用于存储方法调用次数
    private Map<String, Integer> count = new HashMap<>();
 
    // 定义切点,表示拦截com.qcby.springbootdemo.Controller包下的所有方法
    @Pointcut("execution(* com.qcby.springbootdemo.Controller.*.*(..))")
    public void count() {
        System.out.println("切点方法执行");   //声明切点,并不会实际调用
    }
 
    // 环绕通知,在目标方法执行前后进行拦截
    @Around("count()")
    public Object methodExec(ProceedingJoinPoint pjp) throws Throwable {
        System.out.println("方法执行前");
 
        // 获取方法签名信息
        Signature signature = pjp.getSignature();
        String name = signature.getName();
        System.out.println(name);   //-----login
        System.out.println(signature.getDeclaringTypeName());  //---com.qcby.springbootdemo.Controller.LoginController
 
        // 获取方法参数
        Object[] args = pjp.getArgs();
        for (Object arg : args) {
            System.out.println("aop arg:" + arg);
        }
 
        Object result = null;
        System.out.println(signature.toLongString());
        System.out.println();
 
        // 统计方法调用次数,使用方法的签名作为key
        String key = signature.toLongString();
        count.put(key, count.getOrDefault(key, 0) + 1);
 
        // 执行目标方法
        result = pjp.proceed();
        System.out.println("方法执行后");
 
        // 输出访问路由次数
        System.out.println(count);
        return result;
    }
}
复制代码
@Aspect,表示这个类为切面类

// 定义切点,表示拦截com.qcby.springbootdemo.Controller包下的所有方法

@Pointcut("execution(* com.qcby.springbootdemo.Controller.*.*(..))")

public void count() {

System.out.println("切点方法执行"); //声明切点,并不会实际调用

}

复制代码
在拦截count()方法这个切点之后,对原方法------------Controller中访问路由的方法,在执行期间进行环绕,在methodExec方法中写明具体的环绕逻辑
// 环绕通知,在目标方法执行前后进行拦截
@Around("count()")
相关推荐
禁默26 分钟前
深入浅出:AWT的基本组件及其应用
java·开发语言·界面编程
Cachel wood33 分钟前
python round四舍五入和decimal库精确四舍五入
java·linux·前端·数据库·vue.js·python·前端框架
Code哈哈笑35 分钟前
【Java 学习】深度剖析Java多态:从向上转型到向下转型,解锁动态绑定的奥秘,让代码更优雅灵活
java·开发语言·学习
gb421528738 分钟前
springboot中Jackson库和jsonpath库的区别和联系。
java·spring boot·后端
程序猿进阶39 分钟前
深入解析 Spring WebFlux:原理与应用
java·开发语言·后端·spring·面试·架构·springboot
zfoo-framework1 小时前
【jenkins插件】
java
风_流沙1 小时前
java 对ElasticSearch数据库操作封装工具类(对你是否适用嘞)
java·数据库·elasticsearch
颜淡慕潇1 小时前
【K8S问题系列 |19 】如何解决 Pod 无法挂载 PVC问题
后端·云原生·容器·kubernetes
ProtonBase1 小时前
如何从 0 到 1 ,打造全新一代分布式数据架构
java·网络·数据库·数据仓库·分布式·云原生·架构