7000字详解 动态代理(JDK动态代理 CGLIB动态代理)与静态代理

news2024/11/28 6:25:47

代理模式

  • 1. 代理模式 概念
  • 2. 静态代理
  • 3. 动态代理
    • 3.1.JDK动态代理
    • 3.2.CGLIB动态代理
    • 3.3. JDK动态代理和CGLIB动态代理区别
  • 4.静态代理和动态代理区别
  • 5.篇末

1. 代理模式 概念

代理模式是一种设计模式
使用代理对象来替代真实对象,用代理对象去访问目标对象。这样可以保证在不修改目标对象的前提下,还可以增加一些额外的功能,作出扩展。
在这里插入图片描述

代理模式的作用主要是扩展目标对象的功能,在目标对象执行方法的前后,可以自己自定义一些操作。

同时代理模式分为静态代理动态代理

2. 静态代理

静态代理对目标对象的方法增强要手动完成,如果接口增加了新的方法,那么就要更改代码,不够灵活,开发中比较少见静态代理。

JVM层面:
静态代理在编译期讲接口,和接口的实现类,代理类都变成class文件。

静态代理实现:
1. 定义接口A和实现类 (也就是要传输的内容)
2. 创建一个代理类实现接口A (代理类要可以接收传输的内容)
3. 将目标对象注入到代理类中,在代理类的方法中调用目标对象的方法,然后在目标对象方法执行的前后 就可以扩展一些内容。

代码:
1.定义发送内容的接口

package 静态代理;

// 1.定义发送内容的接口    目标对象
public interface Message {
    String send(String message);
}

2.实现发送短信的接口

package 静态代理;

//2. 发送内容接口的实现类      就是目标对象处理传输的内容的处理方法
public class MessageImpl implements Message {
    @Override
    public String send(String message) {
        System.out.println("send message:" + message);
        return message;
    }
}

3.创建代理类并实现接口

package 静态代理;

//3. 代理类 也需要实现接口 且需要将目标对象注入到其中来   代理对象
public class MessageProxy implements Message {

    //目标对象(Message)注入到代理类中
    private final Message Message;

    public MessageProxy(Message message) {
        this.Message = message;
    }

    @Override
    public String send(String message) {
        //方法执行前 添加自定义内容
        System.out.println("before method send()");
        //在代理类中的方法 调用目标类的方法
        Message.send(message);
        //方法执行后 添加自定义内容
        System.out.println("after method send()");
        return null;
    }
}

4.执行代码

public class Main {
    public static void main(String[] args) {
        System.out.println("静态代理");
        //实例化 实现类对象 类型是Message    目标对象
        //Message sendMessage = new MessageImpl();
        MessageImpl sendMessage = new MessageImpl();
        //实例化 代理类对象 完成注入          代理对象
        MessageProxy messageProxy = new MessageProxy(sendMessage);
        //执行方法
        messageProxy.send("hello");
    }
}

在这里插入图片描述

3. 动态代理

对于静态代理来说,动态代理更加灵活,不需要保证每个目标类都有一个代理类,实现接口也不是必须的,可以直接代理实现类。

JVM层面:
动态代理是在运行时动态生成类字节码,加载到JVM中。

举例:
Spring AOP ,RPC框架使用了动态代理。

动态代理对于框架的学习帮助较大

动态代理实现方法很多 ,这篇介绍 JDK动态代理CGLIB动态代理。

3.1.JDK动态代理

JDK动态代理中 InvocationHandler接口Proxy类是重点。

Proxy类中有一个方法:newProxyInstance() 用来生成代理对象。

    public static Object newProxyInstance(ClassLoader loader,
                                          Class<?>[] interfaces,
                                          InvocationHandler h)
        throws IllegalArgumentException
    {
        ......
    }

Proxy类中的newProxyInstance()方法有三个参数:
1.loader:类加载器,用来加载代理对象。
2.interfaces: 被代理类实现的一些接口。
3.h: 实现了InvocationHandler接口的对象。

动态代理还需要实现InvocationHandler接口,自定义处理内容的逻辑,这样我们在代理对象调用一个方法的时候,方法会转发到InvocationHandler接口类中的invoke方法

public interface InvocationHandler {

    /**
     * 当你使用代理对象调用方法的时候实际会调用到这个方法
     */
    public Object invoke(Object proxy, Method method, Object[] args)
        throws Throwable;
}

InvocationHandler接口中的invock方法有三个参数:
**1.proxy:**动态生成的代理类。
**2.method:**与代理类调用的方法对应
**3.args:**method方法的参数

动态代理机制:通过Proxy类的newProxyInstance方法 创建的代理类在调用方法的时候,实际调用的InvocationHandler接口中的Invoke方法。
所以需要在Invoke方法中自定义内容。

JDK动态代理实现:
1.定义一个接口和实现类
2.重写InvocationHandler中的Invoke()方法,用来自定义内容。
3.通过Proxy类的newProxyInstance()方法创建代理对象。

代码:
1.定义发送内容的接口

package JDK动态代理;
//目标对象
public interface SdMessage {
    String send(String message);
}

2.发送内容接口的实现类

package JDK动态代理;
//实现接口的实现类     就是目标对象处理传输的内容的处理方法
public class SdMessageImp implements SdMessage {
    @Override
    public String send(String message) {
        System.out.println("send message:" + message);
        return message;
    }
}

3.重写InvocationHandler中的Invoke方法

package JDK动态代理;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
//代理类调用方法时 调用的是InvocationHandler中的Invoke方法 重写它 自定义
public class MyInvocationHandler implements InvocationHandler {
    /**
     * 代理类中的真实对象
     */
    private final Object target;
    public MyInvocationHandler(Object target) {
        this.target = target;
    }


    //其中的method和代理类调用的方法对应
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        //方法调用前,自定义的内容
        System.out.println("before method() " + method.getName());
        Object result = method.invoke(target,args);
        //方法调用后,自定义的内容
        System.out.println("after method() " + method.getName());
        return result;
    }
}

4.创建代理类

package JDK动态代理;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
//代理类调用方法时 调用的是InvocationHandler中的Invoke方法 重写它 自定义
public class MyInvocationHandler implements InvocationHandler {
    /**
     * 代理类中的真实对象
     */
    private final Object target;
    public MyInvocationHandler(Object target) {
        this.target = target;
    }


    //其中的method和代理类调用的方法对应
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        //方法调用前,自定义的内容
        System.out.println("before method() " + method.getName());
        Object result = method.invoke(target,args);
        //方法调用后,自定义的内容
        System.out.println("after method() " + method.getName());
        return result;
    }
}

5.执行



public class Main {
    public static void main(String[] args) {
        //JDK动态代理  其中的参数是目标对象
        System.out.println("JDK动态代理");
        SdMessage sdMessage = (SdMessage) ProxyFactory.getProxy(new SdMessageImp());
        sdMessage.send("hello");
    }
}

在这里插入图片描述

3.2.CGLIB动态代理

JDK动态代理的缺点是 只可以代理已经实现了接口的实现类。
而CGLIB动态代理可以避免

CGLIB是一个基于ASM的字节码生成库,允许我们在运行时对字节码进行修改和动态生成。CGLIB通过继承方式实现代理,很多开源框架都使用到了CGLIB,例如Spring中的AOP模块中,如果实现了接口那么就采用JDK动态代理,如果没实现,就使用CGLIB动态代理。

CGLIB动态代理MethodInterceptor接口Enhancer类是重点

需要重写MethodInterceptor接口中的Intercept方法,这个方法用来拦截被代理类

public interface MethodInterceptor
extends Callback{
    // 拦截被代理类中的方法
    public Object intercept(Object obj, java.lang.reflect.Method method, Object[] args,MethodProxy proxy) throws Throwable;
}


**MethodInterceptor接口中的Intercept()**方法有4个参数:
1.obj:被代理的对象(也可以叫做需要增强的对象)
2.method:被拦截的方法(也可以叫做需要增强的方法)
3.args:方法的参数
4.proxy:用于调用原始方法

可以通过Enhancer类来动态获取被代理类,当代理类调用方法的时候,实际调用的是MethodInterceptor接口中的Intercept方法。

CGLIB动态代理实现:
1.定义一个类
2.重写MethodInterceptor接口中的Intercept()方法,这个方法用来拦截被代理类的方法,其实也就是拦截到方法以后,会执行intercept方法,和被代理类的方法。
3.通过Enhancer类的create()方法创建代理类。

CGLIB动态代理代码实现:
CGLIB是开源项目需要引入依赖。

<dependency>
  <groupId>cglib</groupId>
  <artifactId>cglib</artifactId>
  <version>3.3.0</version>
</dependency>

1.实现一个发送消息的类

package CGLIB动态代理;

//发送内容的类
public class CglibSdMessage {
    public String send(String message) {
        System.out.println("send message:" + message);
        return message;
    }
}

2.自定义MethodInterceptor(方法拦截器)重写其中的Intercept方法

package CGLIB动态代理;

import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

import java.lang.reflect.Method;

//方法拦截器  重写MethodInterceptor接口中的intercept方法
public class MyMethodInterceptor implements MethodInterceptor {

    /**
     *
     * @param o             被代理的对象(需要增强的对象)
     * @param method        被拦截的方法(需要增强的方法)
     * @param objects       方法的参数
     * @param methodProxy   用于调用原始方法
     * @return
     * @throws Throwable
     */
    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        //方法调用前,自定义的内容
        System.out.println("before method() " + method.getName());
        Object object =  methodProxy.invokeSuper(o,objects);
        //方法调用后,自定义的内容
        System.out.println("after method() " + method.getName());
        return object;
    }
}

3.获取代理类(创建代理类)

package CGLIB动态代理;

import net.sf.cglib.proxy.Enhancer;

//创建代理类 通过Enhancer类的create方法
public class CglibProxyFactory {
    //Class<?> clazz之前接收的是实现接口的实现类 现在接收的是类 泛型
    public static Object getProxy(Class<?> clazz) {
        //创建动态代理类
        Enhancer enhancer = new Enhancer();
        //设置类加载器
        enhancer.setClassLoader(clazz.getClass().getClassLoader());
        //设置被代理类(目标对象)
        enhancer.setSuperclass(clazz);
        //设置方法拦截器
        enhancer.setCallback(new MyMethodInterceptor());
        //创建代理类
        return enhancer.create();
    }
}

4.调用

public class Main {
    public static void main(String[] args) {
        //CGLIB动态代理 其中的参数是被代理类(目标对象)
        System.out.println("JDK动态代理");
        CglibSdMessage cglibSdMessage = (CglibSdMessage) CglibProxyFactory.getProxy(CglibSdMessage.class);
        cglibSdMessage.send("hello");
    }
}

在这里插入图片描述

3.3. JDK动态代理和CGLIB动态代理区别

  • JDK动态代理只能代理已经实现了接口的类,CGLIB动态代理可以直接代理没有实现接口的(CGLIB动态代理通过生成一个被代理类的子类来拦截被代理类中的方法调用,因为是继承所以不能代理被final修饰的类和方法)
  • JDK动态代理效率更高

4.静态代理和动态代理区别

  • 灵活性:动态代理更加的灵活,接口的实现不是必要的,CGLIB动态代理可以直接代理一个类。静态代理只能代理实现了接口的类,并且接口增加新方法的时候,目标对象的代码和代理对象的代码都要修改。
  • JVM:静态代理在编译期就把接口,实现类,代理类变成Class文件,动态代理在运行时动态生成类字节码文件,加载到JVM中。

5.篇末

我在文章中介绍了静态代理和动态代理,包阔了静态代理的代码实现和动态代理中的2钟代理(JDK动态代理和CGLIB动态代理)的代码实现,已经静/动态代理的区别,和两种动态代理的区别。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1256896.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ROS2智能小车基本原理图

我觉得这样意思已经表的很清楚了 这个图很重要&#xff0c;有了这个图&#xff0c;就可以积累每个部分的代码了&#xff0c;如果没有这个图&#xff0c;那么每次都只能是测试&#xff0c;以前的代码都会需要重新写一次。不过第一次训练也许更重要&#xff0c;这也是不可避免的…

使用STM32与MFRC522 IC进行RFID卡的读取与识别(含代码)

利用STM32与MFRC522 IC进行RFID卡的读取和识别&#xff0c;可以实现对RFID卡的读取和获取卡片标识信息。MFRC522 IC是一种高集成度的13.56MHz RFID芯片&#xff0c;常用于门禁系统、物流跟踪和智能支付等领域。下面将介绍如何使用STM32与MFRC522 IC进行RFID卡的读取和识别&…

Mycat实现读写分离

Mycat实现读写分离 Mycat支持MySQL主从复制状态绑定的读写分离机制。这里实现的也是基于MySQL主从复制的读写分离。 MySQL主从复制配置 首先要配置MySQL的主从复制&#xff0c;这里配置的是一主一次从。可以参考下面的文章。 https://blog.csdn.net/wsb_2526/article/detail…

【c++随笔14】虚函数表

【c随笔14】虚函数表 一、虚函数表&#xff08;Virtual Function Table&#xff09;1、定义2、查看虚函数表2.1、 问题&#xff1a;三种类型&#xff0c;包含一个int类型的class、一个int类型的变量、int类型的指针&#xff1a;这三个大小分别是多少呢&#xff1f;2.2、怎么发现…

JVM 内存管理深度剖析

1、JVM 基础知识 1.1 JVM 与操作系统的关系 JVM 能识别 class 后缀的文件&#xff0c;并且能够解析它的指令&#xff0c;最终调用操作系统上的函数&#xff0c;完成指定操作。操作系统并不认识这些 class 文件&#xff0c;是 JVM 将它们翻译成操作系统可识别的机器码&#xf…

【古诗生成AI实战】之二——项目架构设计

[1] 项目架构 在我们深入古诗生成AI项目的具体实践之前&#xff0c;让我们首先理解整个项目的架构。本项目的代码流程主要分为三个关键阶段&#xff1a; 1、数据处理阶段&#xff1b;   2、模型训练阶段&#xff1b;   3、文本生成阶段。 第一步&#xff1a;在数据处理阶段…

WordPress无需插件禁用WP生成1536×1536和2048×2048尺寸图片

我们在使用WordPress上传图片媒体文件的时候&#xff0c;是不是看到媒体库中有15361536和20482048的图片文件&#xff0c;当然这么大的文件会占用我们的服务器空间&#xff0c;如何禁止掉呢&#xff1f; function remove_default_image_sizes( $sizes) {unset( $sizes[1536x15…

Gee教程1.HTTP基础

标准库启动web服务 Go语言内置了 net/http库&#xff0c;封装了HTTP网络编程的基础的接口。这个Web 框架便是基于net/http的。我们先回顾下这个库的使用。 package mainimport ("fmt""log""net/http" )func main() {//可以写成匿名函数(lambda…

Java 注解在 Android 中的使用场景

Java 元注解有 5 种&#xff0c;常用的是 Target 和 Retention 两个。 其中 Retention 表示保留级别&#xff0c;有三种&#xff1a; RetentionPolicy.SOURCE - 标记的注解仅保留在源码级别中&#xff0c;并被编译器忽略RetentionPolicy.CLASS - 标记的注解在编译时由编译器保…

树状数组 / pbds解法 E2. Array Optimization by Deque

Problem - 1579E2 - Codeforces Array Optimization by Deque - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 树状数组解法 将 a i a_i ai​插入到队头&#xff0c;贡献为&#xff1a;原队列中所有比 a i a_i ai​小的数的数量将 a i a_i ai​插入到队尾&#xff0c;贡献为&a…

深信服超融合一体机提示:内存ECC

PS&#xff1a;此事件分享主要来源于季度巡检时发现的超融合一体机红灯闪烁异常&#xff0c;接入IPMI端口查看日志发现持续提示内存ECC&#xff1b; 因为是只有3.05这一天发现了有这个告警的提示&#xff0c;所以当时清除了日志以后重启了BMC服务就解决了&#xff1b;但是如果清…

常见树种(贵州省):021冬青、连香树、白辛树、香合欢、云贵鹅耳枥、肥牛树、杜英、格木、黄连木、圆果化香树、南天竹

摘要&#xff1a;本专栏树种介绍图片来源于PPBC中国植物图像库&#xff08;下附网址&#xff09;&#xff0c;本文整理仅做交流学习使用&#xff0c;同时便于查找&#xff0c;如有侵权请联系删除。 图片网址&#xff1a;PPBC中国植物图像库——最大的植物分类图片库 一、冬青 …

MyBatis插入操作返回主键报错问题记录

一开始用直接传参数的方法写的插入操作 StudentMapper.java接口 Integer insertStudent(Param("sname") String name,Param("sage") int age); 然后在网上搜了返回主键的方法 StudentMapper.xml: <insert id"insertStudent" useGenerat…

简易版王者荣耀

所有包和类 GameFrame类 package newKingOfHonor;import java.awt.*; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import java.awt.event.KeyAdapter; import java.awt.event.KeyEvent; import java.io.File; import java.util.ArrayList;im…

【操作系统】Linux操作系统中命令行参数与环境变量

本篇要分享的内容关于Linux新操作系统中命令行参数。 命令行参数本质上还是作为以后的环境变量的基础来学习的&#xff0c;所以在接触更高难度的内容之前先学习基础。 以下为本篇目录 目录 1.main函数的参数&#xff1f; 2.命令行解释器意义 3.环境变量 3.1.PATH环境变量…

手把手教会你--渗透实战--Hack The Box-Starting Point-Meow--持续更新

有什么问题&#xff0c;请尽情问博主&#xff0c;QQ群796141573 前言 前言 请务必跟着博主复现一遍 参考&#xff1a; Hack The Box-Starting Point-Meow

工业级 S25HS01GTDPBHV030 NOR闪存,L9305EP汽车级驱动器IC,LMK03318RHSR时钟发生器,PLL(中文资料)

一、工业级 S25HS01GTDPBHV030 Semper™ NOR闪存 S25HS01GT SEMPER™ NOR Flash闪存系列是英飞凌高性能、安全而可靠的 NOR Flash解决方案。 它集成了适用于汽车、工业、通信等广泛应用的关键安全功能。 凭借 SEMPER™ NOR Flash闪存&#xff0c;英飞凌推出了业界首款符合 ASI…

设计模式—接口隔离原则(ISP)

1.背景 2002 年罗伯特C.马丁给“接口隔离原则”的定义是&#xff1a;客户端不应该被迫依赖于它不使用的方法&#xff08;Clients should not be forced to depend on methods they do not use&#xff09;。该原则还有另外一个定义&#xff1a;一个类对另一个类的依赖应该建立…

C#学习-9课时

P11 IF判断(上) P11 IF判断(中 ) bool→true or false&#xff1b; 为&#xff1a;变量赋值 为&#xff1a;等于(判断) !为&#xff1a;≠ 优先级&#xff1a;大于 using System; using System.Collections.Generic; using System.Linq; using System.Text; usin…

USB总线-Linux内核USB3.0 Hub驱动分析(十四)

1.概述 USB Hub提供了连接USB主机和USB设备的电气接口。USB Hub拥有一个上行口&#xff0c;至少一个下行口&#xff0c;上行口连接上一级的Hub的下行口或者USB主机&#xff0c;连接主机的为Root Hub&#xff0c;下行口连接下一级Hub的上行口或者USB设备。经过Hub的扩展&#x…