代理模式--静态代理和动态代理

news2025/1/21 3:02:12

1.代理模式

定义:代理模式就是代替对象具备真实对象的功能,并代替真实对象完成相应的操作并且在不改变真实对象源代码的情况下扩展其功能,在某些情况下,⼀个对象不适合或者不能直接引⽤另⼀个对象,⽽代理对象可以在客户端和⽬标对象之间起到中介的作⽤

使用代理模式可以降低系统的耦合性,扩展性好,并且可以起到保护目标对象的作用
例如:我们平时租房的过程,租房中介就相当于代理类
代理模式分为静态代理和动态代理

2.静态代理

静态代理实现步骤:

  1. 定义⼀个接⼝及其实现类(目标类);
  2. 创建⼀个代理类同样实现这个接⼝(继承同一个接口的原因就是,代理类需要拥有和目标类同样的方法这样才能代理)
  3. 将⽬标对象注⼊进代理类,然后在代理类的对应⽅法调⽤⽬标类中的对应⽅法。
public interface IRentHouse {
    void rent();
}

public class RentHouse implements IRentHouse{
    @Override
    public void rent() {
        System.out.println("租户租房子");
    }
}

public class IntermediaryProxy implements IRentHouse{
    private IRentHouse iRentHouse;
    public IntermediaryProxy(IRentHouse iRentHouse) {
        this.iRentHouse = iRentHouse;
    }
    @Override
    public void rent() {
        System.out.println("交中介费");
        iRentHouse.rent();
        System.out.println("租房子后中介负责维护管理");

    }
}
/**
 * 测试类
 */
public class Test {
    public static void main(String[] args) {
        // 定义租房
        IRentHouse iRentHouse = new RentHouse();
        // 定义中介
        IRentHouse proxy = new IntermediaryProxy(iRentHouse);
        // 租房
        proxy.rent();
    }
}

运行结果如下:
在这里插入图片描述
静态代理有很多缺点,实际应用场景非常少,几乎不用
对目标对象的每个方法的增强都是手动完成的,非常不灵活(比如接口中一旦新增方法,目标对象和代理对象都要修改),且麻烦(需要对每个目标类都单独写一个代理类)

3.动态代理

相比于静态代理来说,动态代理更加灵活,不需要针对每个目标类都单独创建一个代理类,也不需要我们必须实现接口
动态代理允许使用一种方法的单个类(代理类),为具有任意数量方法的任意类(目标类)的多个方法提供服务,看到这句话,是不是联想到动态代理的实现与Java反射机制密不可分

Java反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法,对于任意一个对象,都能够调用它的任意一个属性和方法,这种动态获取的信息以及动态调用对象的方法的功能称之为Java语言的反射机制

从 JVM ⻆度来说,动态代理是在运⾏时动态⽣成类字节码,并加载到 JVM 中
的。说到动态代理,不得不提的是Spring AOP,它的实现依赖了动态代理

代理类的两个作用
1.添加增强方法,2.调用目标类

1.jdk动态代理(接口代理)

在 Java 动态代理机制中java.long.reflect包中的 InvocationHandler 接⼝和 Proxy 类是核⼼

实际上就是在内存中生产一个对象,该对象实现了指定的目标对象的所有接口,代理对象和目标对象是兄弟关系,
jdk自带动态代理技术,需要使用一个静态方法来创建代理对象,他需要目标对象必须实现接口,生产的代理对象和目标对象都实现同一个接口

JDK 动态代理类使⽤步骤:

  1. 定义⼀个接⼝及其实现类;
  2. ⾃定义 InvocationHandler 并重写invoke⽅法,在 invoke ⽅法中我们会调⽤ 原⽣⽅法(被代理类的⽅法)并⾃定义⼀些处理逻辑;
  3. 通过 Proxy.newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h) ⽅法创建代理对象;

1.定义JDK动态代理类

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

// JDK 动态代理类
public class JDKInvocationHandler implements InvocationHandler {

    //⽬标对象即就是被代理对象
    private Object target;

    public JDKInvocationHandler(Object target) {
        this.target = target;
    }

    /**
     *
     * @param proxy 代理对象
     * @param method 代理方法
     * @param args 参数
     *
     * @return
     * @throws Throwable
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        // 1234就是一些增强方法
        //1.安全检查
        System.out.println("安全检查");
        //2.记录⽇志
        System.out.println("记录⽇志");
        //3.时间统计开始
        System.out.println("记录开始时间");
        //通过反射调⽤被代理类的⽅法
        Object retVal = method.invoke(target, args);
        //4.时间统计结束
        System.out.println("记录结束时间");
        return retVal;
    }
}

2.创建⼀个代理对象并使用

public class Main {
    public static void main(String[] args) {
        // 代理对象
        PayService target= new AliPayService();
        // 静态的是已经写好了的
        // 动态的创建⼀个代理类:通过被代理类、被代理实现的接⼝、⽅法调⽤处理器来创建
        PayService proxy = (PayService) Proxy.newProxyInstance(
                // 通过目标类的getClassLoader
                target.getClass().getClassLoader(),
                // 被代理类实现的一些接口
                new Class[]{PayService.class},
                // 实现了InvocationHandler接口的对象
                new JDKInvocationHandler(target)
        );
        proxy.pay();
    }

}

Proxy 类中使⽤频率最⾼的⽅法是:newProxyInstance() ,这个⽅法主要⽤来⽣成⼀个代理对象

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

这个⽅法⼀共有 3 个参数:

  1. loader :类加载器,⽤于加载代理对象。
  2. interfaces : 被代理类实现的⼀些接⼝;
  3. h : 实现了 InvocationHandler 接⼝的对象;

运行main方法
在这里插入图片描述
JDK 动态代理有⼀个最致命的问题是其只能代理实现了接⼝的类,为了解决这个问题,我们可以⽤ CGLIB 动态代理机制来避免

CGLIB(Code GenerationLibrary)是⼀个基于ASM的字节码⽣成库,它允许我们在运⾏时对字节码进⾏修改和动态⽣成。CGLIB通过继承⽅式实现代理。很多知名的开源框架都使⽤到了CGLIB, 例如 Spring 中的 AOP 模块中:如果⽬标对象实现了接⼝,则默认采⽤JDK 动态代理,否则采⽤ CGLIB 动态代理。

在 CGLIB 动态代理机制中 MethodInterceptor 接⼝和 Enhancer 类是核⼼

2.CGLIB 动态代理类使⽤步骤

  1. 定义⼀个类;
  2. ⾃定义 MethodInterceptor 并重写 intercept ⽅法,intercept ⽤于拦截增强
    被代理类的⽅法,和 JDK 动态代理中的 invoke ⽅法类似;
  3. 通过 Enhancer 类的 create()创建代理类

1.添加依赖

和JDK 动态代理不同, CGLIB(Code Generation Library) 实际是属于⼀个开源项⽬,如果你要使⽤它的话,需要⼿动添加相关依赖

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

2.⾃定义 MethodInterceptor(⽅法拦截器)

public class CGLIBInterceptor implements MethodInterceptor {
 //被代理对象
 private Object target;
 public CGLIBInterceptor(Object target){
 this.target = target;
 }
 @Override
 public Object intercept(Object o, Method method, Object[] args, Method
Proxy methodProxy) throws Throwable {
 //1.安全检查
 System.out.println("安全检查");
 //2.记录⽇志
 System.out.println("记录⽇志");
 //3.时间统计开始
 System.out.println("记录开始时间");
 //通过cglib的代理⽅法调⽤
 Object retVal = methodProxy.invoke(target, args);
 //4.时间统计结束
 System.out.println("记录结束时间");
 return retVal;
 }
}

3.创建代理类, 并使⽤

public static void main(String[] args) {
 PayService target= new AliPayService();
 PayService proxy= (PayService) Enhancer.create(target.getClass(),ne
w CGLIBInterceptor(target));
 proxy.pay();
 }

你需要⾃定义 MethodInterceptor 并重写 intercept ⽅法,intercept ⽤于拦截增强被代理类的⽅法

public interface MethodInterceptor
extends Callback{
 // 拦截被代理类中的⽅法
 public Object intercept(Object obj, java.lang.reflect.Method method, Ob
ject[] args,MethodProxy proxy) throws Throwable;
}
  • obj : 被代理的对象(需要增强的对象)
  • method : 被拦截的⽅法(需要增强的⽅法)
  • args : ⽅法⼊参
  • proxy : ⽤于调⽤原始⽅法

3.JDK 动态代理和 CGLIB 动态代理对⽐:

  • JDK 动态代理只能代理实现了接⼝的类或者直接代理接⼝,⽽ CGLIB 可以代理未实现任何接⼝的类
  • CGLIB动态代理是通过⽣成⼀个被代理类的⼦类来拦截被代理类的⽅法调⽤,因此不能代理声明为 final

性能: ⼤部分情况都是 JDK 动态代理更优秀,随着 JDK 版本的升级,这个优势更加明显

Spring代理选择

  1. proxyTargetClass 为false, ⽬标实现了接⼝, ⽤jdk代理
  2. proxyTargetClass 为false, ⽬标未实现接⼝, ⽤cglib代理
  3. proxyTargetClass 为true, ⽤cglib代理

下篇见~
在这里插入图片描述

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

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

相关文章

华为OD机试真题 Java 实现【最长公共后缀】【2023 B卷 100分】,等于白送

目录 专栏导读一、题目描述二、输入描述三、输出描述四、解题思路五、Java算法源码六、效果展示1、输入2、输出3、说明4、再输入5、再输出 七、机考攻略 华为OD机试 2023B卷题库疯狂收录中&#xff0c;刷题点这里 专栏导读 本专栏收录于《华为OD机试&#xff08;JAVA&#xff…

【Golang】Golang进阶系列教程--Go 语言数组和切片的区别

文章目录 前言数组声明以及初始化函数参数 切片声明以及初始化函数参数 总结 前言 在 Go 语言中&#xff0c;数组和切片看起来很像&#xff0c;但其实它们又有很多的不同之处&#xff0c;这篇文章就来说说它们到底有哪些不同。 数组和切片是两个常用的数据结构。它们都可以用…

LBERT论文详解

论文地址&#xff1a;https://arxiv.org/abs/2105.07148 代码地址&#xff1a;https://github.com/liuwei1206/LEBERT 模型创新 LEBRT采用句子中的词语对&#xff08;论文中称为Char-Word Pair&#xff09;的特征作为输入作者设计Lexicon adapter&#xff0c;在BERT的中间某一…

Codeforces Round 839 (Div. 3)E题解

文章目录 [Permutation Game](https://codeforces.com/contest/1772/problem/E)问题建模问题分析1.分析一个玩家想要获胜的关键2.分析阻塞元素的类别3.分析阻塞元素的类别对于局面的影响代码 Permutation Game 问题建模 给定一个长度为n的排列&#xff0c;排列的每个元素都被阻…

CentOS 7安装PostgreSQL 15版本数据库

目录 一、何为PostgreSQL&#xff1f; 二、PostgreSQL安装 2.1安装依赖 2.2 执行安装 2.3 数据库初始化 2.4 配置环境变量 2.5 创建数据库 2.6 配置远程 2.7 测试远程 三、常用命令 四、用户创建和数据库权限 一、何为PostgreSQL&#xff1f; PostgreSQL是以加州大学…

DGNN Survey

Dynamic Graph Definition G ( V , E , X ) G (V, E, X) G(V,E,X) V v 1 , v 2 , . . . , v m V {v_1, v_2, ..., v_m} Vv1​,v2​,...,vm​ E e i , j E {e_{i, j}} Eei,j​ , e i , j ( v i , v j , f i , j ) e_{i,j} (v_i, v_j, f_{i,j}) ei,j​(vi​,vj​,fi,j​…

M1/M2 通过VM Fusion安装Win11 ARM,解决联网和文件传输

前言 最近新入了Macmini M2&#xff0c;但是以前的老电脑的虚拟机运行不起来了。&#x1f605;&#xff0c;实际上用过K8S的时候&#xff0c;会发现部分镜像也跑不起来&#xff0c;X86的架构和ARM实际上还是有很多隐形兼容问题。所以只能重新安装ARM Win11&#xff0c;幸好微软…

【初阶C语言】整数比大小

各位大佬的光临已是上上签 在C语言刷题过程中&#xff0c;一定遇到过很多比大小的题目&#xff0c;那么本节就专门介绍比大小的方法&#xff0c;若大佬们还有更优解&#xff0c;欢迎补充呀&#xff01; 本节讲解的方法主要有三种&#xff1a;1.条件判断 2.三目操作符 3.函数调…

JA64 1+2+3+...n

一、题目 求123...n_牛客题霸_牛客网 二、代码 1.使用静态成员变量构造函数 class SUM {private:static int _i;static int _ret;public:SUM(){_ret _ret _i;_i;}static int GetRet(){return _ret;} }; int SUM::_i1; int SUM::_ret0;class Solution { public:int Sum_So…

暴力求解--完数个数(等于本身之外的因子之和)

找出10000以内的自然数中的所有完数&#xff0c;并统计找到的完数个数。 #include<stdio.h> int main() {//找到10000以内所有的完数&#xff08;等于恰好等于它本身之外的因子之和&#xff09;&#xff0c;并统计完数个数。int n,i,s,count0;printf("找到的所有完…

jsonp 实现跨域 同时也是一个 webflux 的demo 示例

文章目录 核心原理代码html服务端 &#xff08;java 为例子&#xff09;服务端目录结构 核心原理 前端&#xff1a; 使用js 创建 script 标签&#xff0c;将请求地址&#xff0c;放到其src 中&#xff0c;并将 script 标签追加到文档流&#xff1b;后端&#xff1a;根据约定好…

Latex好看的引用(文献,url, 文内引用)

强迫症实锤了&#xff0c;完全符合本人审美&#xff01;&#xff01;&#xff01; \usepackage{hyperref} \hypersetup{ hidelinks, colorlinkstrue, linkcolorIndigo, urlcolorDeepSkyBlue4, citecolorIndigo }基本还原了 哼&#xff0c;欺负老子色彩妹那么敏感是吧&…

WIZnet W51000S-EVB-PICO 入门教程(一)

概述 W5100S-EVB-Pico是基于树莓派RP2040和全硬件TCP/IP协议栈控制器W5100S的微控制器开发板-基本上与树莓派Pico板相同&#xff0c;但通过W5100S芯片增加了以太网功能。 W5100S-EVB-Pico特点 RP2040规格参数 双核Arm Cortex-M0 133MHz264KB 高速SRAM和2MB板载内存通过…

Docker复杂命令便捷操作

启动所有状态为Created的容器 要启动所有状态为"created"的Docker容器&#xff0c;可以使用以下命令&#xff1a; docker container start $(docker container ls -aq --filter "statuscreated")上述命令执行了以下步骤&#xff1a; docker container l…

【Linux】-进程概念及初始fork

&#x1f496;作者&#xff1a;小树苗渴望变成参天大树&#x1f388; &#x1f389;作者宣言&#xff1a;认真写好每一篇博客&#x1f4a4; &#x1f38a;作者gitee:gitee✨ &#x1f49e;作者专栏&#xff1a;C语言,数据结构初阶,Linux,C 动态规划算法&#x1f384; 如 果 你 …

C++ 拷贝构造函数

拷贝构造函数是一种特殊的构造函数&#xff0c;具有一般构造函数的所有特性&#xff0c;其形参是本类的对象的引用。其作用是使用一个已经存在的对象&#xff08;由拷贝构造函数的参数指定&#xff09;&#xff0c;去初始化同类的一个新对象。 如果程序员没有定义类的拷贝构造函…

自动驾驶感知系统--惯性导航定位系统

惯性导航定位 惯性是所有质量体本身的基本属性&#xff0c;所以建立在牛顿定律基础上的惯性导航系统&#xff08;Inertial Navigation System,INS&#xff09;(简称惯导系统)不与外界发生任何光电联系&#xff0c;仅靠系统本身就能对车辆进行连续的三维定位和三维定向。卫星导…

Ubuntu-文件和目录相关命令一

&#x1f52e;linux的文件系统结构 ⛳目录结构及目录路径 &#x1f9e9;文件系统层次结构标准FHS Filesystem Hierarchy Standard(文件系统层次结构标准&#xff09; Linux是开源的软件&#xff0c;各Linux发行机构都可以按照自己的需求对文件系统进行裁剪&#xff0c;所以众多…

MyBatisPlus从入门到精通-3

紧接着上一篇的查询 接下来的重点介绍增删改操作了 Insert id&#xff08;主键&#xff09;生成策略 前面的案列中我们没有指定id字段 但是它是生成了一个很长的id&#xff0c;并不是我们数据表定义自增 这是Mp内部算法出来的一个值 其实根据不同应用场景&#xff0c;应该使…

抖音SEO源代码的部署与搭建技巧详解

抖音SEO源代码的部署与搭建是一项重要的技术&#xff0c;促进了抖音的发展。在此&#xff0c;我将为大家详细介绍抖音SEO源代码的部署与搭建技巧。 首先&#xff0c;我们需要了解抖音SEO源代码的含义。SEO源代码是搜索引擎优化的核心&#xff0c;它是用于帮助搜索引擎更好地理解…