【关于Spring那些事】——带你了解Spring AOP的代理模式

简介: 代理摸式是一种设计横式,提供了对目标对象额外的访问开式,即通过代理对象来实现对目标对象的访问,能够在不修改原目标对象的前提下提供额外的功能操作,实现对目标对象的功能扩展。简单来说,代理模式就是设置一个中间代理来控制访问原目标对象,从而增加原对象的功能和简化访问方式。

一、静态代理

静态代理,代理类和被代理的类实现了同样的接口,只能代理特定的类。
  1. 定义一个接口
public interface Service {
    public void delete();
    public void update();
    public void insert();
}
  1. 被代理类
public class StudentServiceImpl implements Service {
    @Override
    public void delete() {
        System.out.println("删除学生信息!!!");
    }

    @Override
    public void update() {
        System.out.println("修改学生信息!!!");
    }

    @Override
    public void insert() {
        System.out.println("添加学生信息!!!");
    }
}
  1. 代理类
public class StaticProxy implements Service {

    private Service service;

    public StaticProxy(Service service) {
        this.service = service;
    }

    public void delete() {
        open();
        service.delete();
        commit();
    }

    public void update() {
        open();
        service.update();
        commit();
    }

    public void insert() {
        open();
        service.insert();
        commit();
    }

    public void open(){
        System.out.println("开启事务!!!");
    }
    public void commit(){
        System.out.println("提交事务!!!");
    }

}

4.测试类

public class Test {
    public static void main(String[] args) {
        //被代理类
        StudentServiceImpl student = new StudentServiceImpl();
        //被代理类的代理对象
        StaticProxy proxy = new StaticProxy(student);
        //通过代理对象调用被代理类的方法
        proxy.delete();
    }
}

运行结果:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-1tskzJql-1650096958348)(img_1.png)]

被代理类只需负责自己特定的业务,而代理类则负责业务的扩展,比如执行被代理类的方法前要开启事务,执行之后要提交事务,我们就不需要给每个方法都添加开启事务和提交事务。
代理类就会负责完成这些工作。

静态代理的优缺点:

  1. 优点:
  • 被代理类只需负责核心业务
  • 业务逻辑的扩展更加方便
  • 通用代码放到代理类中,提高了代码的复用性
  1. 缺点:

    • 被代理类太多,就会导致工作量变大,开发效率降低

二、动态代理

动态代理,由AOP框架动态生成的一的对象,对象可以作为目标对象使用。
动态代理有两种方式:

  • JDK动态代理
  • CGLIB代理

2.1 JDK动态代理

基于接口的动态代理,只能为实现了接口的类动态代理对象

  • 创建一个接口并创建一个它的实现类并重写它的方法。
  • 创建一个类实现InvocationHandler接口,并重写invoke方法
public class JdkDynamicProxy implements InvocationHandler {
    //被代理的对象
    private Object object;

    public JdkDynamicProxy(Object object) {
        this.object = object;
    }
    //产生代理对象,返回代理对象
    public Object getProxy(){
        //1.获取被代理对象的类加载器
        ClassLoader classLoader = object.getClass().getClassLoader();
        //2.获取被代理对象实现的所有接口
        Class<?>[] interfaces = object.getClass().getInterfaces();
        //3.创建代理对象
        //classloader:类加载器来定义代理类
        //interfaces:代理类实现的接口列表
        //this:调度方法调用的调用处理函数
        Object o = Proxy.newProxyInstance(classLoader, interfaces,this);
        return o;
    }
    //处理代理实例,返回结果
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        log(method.getName());
        Object invoke = method.invoke(object,args);
        return invoke;
    }
    //定义一个打印日志的方法
    public void log(String msg){
        System.out.println("执行了"+ msg +"方法!");
    }
}
  • 测试
public class DynamicProxyTest {
    public static void main(String[] args) {
        //创建被代理类对象
        StudentServiceImpl studentService = new StudentServiceImpl();
        //创建代理对象,产生的代理对象可以强转成被代理对象实现的接口类型
        JdkDynamicProxy jdkDynamicProxy = new JdkDynamicProxy(studentService);
        Service proxy = (Service) jdkDynamicProxy.getProxy();
        //使用代理对象调方法,不会执行调用的方法,而是进入到创建代理对象时指定的invoke方法
        //调用的方法作为一个参数,传给invoke方法
        proxy.delete();
    }
}

运行结果:
在这里插入图片描述

2.2 CGLib代理

使用JDK动态代理的对象必须是实现了一个或多个接口的,如果要对没有实现接口的类创建代理对象,就要使用CGLIB代理。
基于类的动态代理—CGlib
  • CGLib是一个高性能开源的代码生成包,因在Spring的核心包中已包含CGLib所需要的包,所以不再需要添加依赖。
  • 创建一个StudentServiceImpl 类,并添加增删改方法。
  • 创建一个类,实现MethodInterceptor接口,并重写intercept方法。
/**
 * CGLIB代理
 */
public class CGLibDynamicProxy implements MethodInterceptor {

    private Object object;

    public CGLibDynamicProxy(Object object) {
        this.object = object;
    }
    //创建并返回代理对象
    //该代理对象是通过被代理类的子类来创建的
    public Object getProxy(){
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(object.getClass());
        enhancer.setCallback(this);
        return enhancer.create();
    }

    @Override
    public Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
        log(method.getName());
        Object invoke = method.invoke(object, args);
        return invoke;
    }

    //定义一个打印日志的方法
    public void log(String msg){
        System.out.println("执行了"+ msg +"方法!");
    }
}
  • 测试
public class DynamicProxyTest {
    public static void main(String[] args) {
        //创建被代理类对象
        StudentServiceImpl studentService = new StudentServiceImpl();
        //创建代理对象,产生的代理对象可以强转成被代理类类型
        CGLibDynamicProxy cgLibDynamicProxy = new CGLibDynamicProxy(studentService);
        StudentServiceImpl proxy = (StudentServiceImpl) cgLibDynamicProxy.getProxy();
        //使用代理对象调方法,不会执行调用的方法,而是进入到创建代理对象时指定的intercept方法
        //将调用的方法以及方法中的参数传给intercept方法
        proxy.insert();
    }
}

运行结果:
在这里插入图片描述
动态代理的优点:

  • 被代理类只需负责核心业务;
  • 业务逻辑的扩展更加方便;
  • 通用代码放到代理类中,提高了代码的复用性;
  • 一个动态代理 , 一般代理某一类业务;
  • 一个动态代理可以代理多个类,代理的是接口。
相关文章
|
2月前
|
Java
Spring5入门到实战------9、AOP基本概念、底层原理、JDK动态代理实现
这篇文章是Spring5框架的实战教程,深入讲解了AOP的基本概念、如何利用动态代理实现AOP,特别是通过JDK动态代理机制在不修改源代码的情况下为业务逻辑添加新功能,降低代码耦合度,并通过具体代码示例演示了JDK动态代理的实现过程。
Spring5入门到实战------9、AOP基本概念、底层原理、JDK动态代理实现
|
3天前
|
存储 缓存 Java
Spring高手之路23——AOP触发机制与代理逻辑的执行
本篇文章深入解析了Spring AOP代理的触发机制和执行流程,从源码角度详细讲解了Bean如何被AOP代理,包括代理对象的创建、配置与执行逻辑,帮助读者全面掌握Spring AOP的核心技术。
11 3
Spring高手之路23——AOP触发机制与代理逻辑的执行
|
1月前
|
设计模式 Java 测试技术
spring复习04,静态代理动态代理,AOP
这篇文章讲解了Java代理模式的相关知识,包括静态代理和动态代理(JDK动态代理和CGLIB),以及AOP(面向切面编程)的概念和在Spring框架中的应用。文章还提供了详细的示例代码,演示了如何使用Spring AOP进行方法增强和代理对象的创建。
spring复习04,静态代理动态代理,AOP
|
3月前
|
Java Spring
在Spring Boot中使用AOP实现日志切面
在Spring Boot中使用AOP实现日志切面
|
13天前
|
Java 编译器 Spring
Spring AOP 和 AspectJ 的区别
Spring AOP和AspectJ AOP都是面向切面编程(AOP)的实现,但它们在实现方式、灵活性、依赖性、性能和使用场景等方面存在显著区别。‌
28 2
|
21天前
|
Java Spring 容器
Spring IOC、AOP与事务管理底层原理及源码解析
【10月更文挑战第1天】Spring框架以其强大的控制反转(IOC)和面向切面编程(AOP)功能,成为Java企业级开发中的首选框架。本文将深入探讨Spring IOC和AOP的底层原理,并通过源码解析来揭示其实现机制。同时,我们还将探讨Spring事务管理的核心原理,并给出相应的源码示例。
74 9
|
13天前
|
XML Java 数据格式
Spring的IOC和AOP
Spring的IOC和AOP
32 0
|
1月前
|
Java 数据库连接 数据库
Spring基础3——AOP,事务管理
AOP简介、入门案例、工作流程、切入点表达式、环绕通知、通知获取参数或返回值或异常、事务管理
Spring基础3——AOP,事务管理
|
2月前
|
XML Java 数据格式
Spring5入门到实战------11、使用XML方式实现AOP切面编程。具体代码+讲解
这篇文章是Spring5框架的AOP切面编程教程,通过XML配置方式,详细讲解了如何创建被增强类和增强类,如何在Spring配置文件中定义切入点和切面,以及如何将增强逻辑应用到具体方法上。文章通过具体的代码示例和测试结果,展示了使用XML配置实现AOP的过程,并强调了虽然注解开发更为便捷,但掌握XML配置也是非常重要的。
Spring5入门到实战------11、使用XML方式实现AOP切面编程。具体代码+讲解
|
2月前
|
缓存 Java 开发者
Spring高手之路22——AOP切面类的封装与解析
本篇文章深入解析了Spring AOP的工作机制,包括Advisor和TargetSource的构建与作用。通过详尽的源码分析和实际案例,帮助开发者全面理解AOP的核心技术,提升在实际项目中的应用能力。
37 0
Spring高手之路22——AOP切面类的封装与解析