Java Review - 关于代理的二三事儿

文章目录

  • Pre
  • 概述
  • 静态代理
    • 概述
    • Code
  • 动态代理
    • 概述
    • 实现方式一 - JDK代理或接口代理
      • 概述
      • Code
    • 实现方式二 - CGLib 子类代理 (Code Generation Library)
      • 概述
      • pom依赖
      • Code

在这里插入图片描述


Pre

Java-JDK动态代理

Java-CGLib动态代理


概述

代理模式是一种结构型设计模式,其目的是为其他对象提供一种代理以控制对这个对象的访问。

在 Java 中,代理模式有两种形式:

  • 静态代理
  • 动态代理

在代理模式中,代理类和目标类之间有一个抽象接口,代理类实现了这个接口,而目标类则实现了具体的业务逻辑。通过代理类调用目标类,可以实现对目标类的访问控制、权限检查、审计等功能。

代理模式在 Java 中应用广泛,例如可以使用代理模式来实现对网络请求的缓存、访问控制、限流等功能。


静态代理

在这里插入图片描述

概述

静态代理是一种在编译时生成代理类的方式。在静态代理中,代理类和委托类的关系在运行前就确定了。静态代理模式的实现需要手动编写代理类,并使用代理类来调用目标类。

静态代理是指在编译时就已经确定了代理类和被代理类的关系,代理类和被代理类都要实现同一个接口或者继承同一个父类。在静态代理中,代理类负责调用被代理类的方法,并在方法调用前后进行一些额外的处理

静态代理的优点是简单易懂,容易实现,缺点是需要为每个被代理类编写一个代理类,如果被代理类过多,会导致代码冗长和维护困难。


Code

在这里插入图片描述

package com.artisan.proxy.static_proxy;/*** @author artisan*/
public interface Subject {void bussiness();
}
package com.artisan.proxy.static_proxy;/*** @author 小工匠* @version 1.0* @mark: show me the code , change the world*/
public class RealSubject implements Subject {@Overridepublic void bussiness() {System.out.println("RealSubject bussiness  Logic ");}
}
package com.artisan.proxy.static_proxy;/*** @author 小工匠* @version 1.0* @mark: show me the code , change the world* @desc: 代理类也需要实现同一个接口, 并持有真正的业务对象*/
public class ProxySubject implements Subject {/*** 持有真正的对象*/private RealSubject realSubject;/*** 传入要代理的对象, 通过构造函数实例化代理对象** @param realSubject*/public ProxySubject(RealSubject realSubject) {this.realSubject = realSubject;}@Overridepublic void bussiness() {// 代理前的业务逻辑System.out.println("ProxySubject before request");// 真正的业务处理逻辑realSubject.bussiness();// 代理后业务逻辑System.out.println("ProxySubject after request");}
}
package com.artisan.proxy.static_proxy;/*** @author 小工匠* @version 1.0* @mark: show me the code , change the world*/
public class StaticProxyDemo {public static void main(String[] args) {// 真正的处理对象RealSubject realSubject = new RealSubject();// 传入真正的处理对象,初始化代理对象ProxySubject proxySubject = new ProxySubject(realSubject);// 通过代理类调用实现了同一个接口或者继承了同一个父类的方法,以便织入代理逻辑,且不影响原来的方法的逻辑proxySubject.bussiness();}
}

在这里插入图片描述


动态代理

概述

动态代理是一种在运行时生成代理类的方式。在动态代理中,代理类和委托类的关系是在运行时确定的。动态代理模式的实现可以使用 Java 提供的 Proxy 类和 InvocationHandler 接口来实现 或者 CGLib动态代理。


实现方式一 - JDK代理或接口代理

在这里插入图片描述

概述

动态代理是指在运行时动态生成代理类,不需要为每个被代理类编写一个代理类。在动态代理中,代理类实现了InvocationHandler接口,通过反射机制调用被代理类的方法,并在方法调用前后进行一些额外的处理

动态代理的优点是可以动态生成代理类,不需要为每个被代理类编写一个代理类,缺点是实现较为复杂。

Java动态代理是一种在运行时创建代理类的机制,它允许在不提前知道代理类的具体类型的情况下,动态地创建一个代理对象来代替原始类。相比于静态代理,动态代理更加灵活,可以代理任意的接口类型,不需要为每个被代理的类编写专门的代理类,而是通过Java的反射机制在运行时动态生成代理类。

动态代理主要使用java.lang.reflect.Proxy类和java.lang.reflect.InvocationHandler接口来实现。动态代理又被称为JDK代理或接口代理。

Code

在这里插入图片描述

package com.artisan.proxy.dynamic_jdk_proxy;/*** @author artisan*/
public interface Subject {void request();String handle(String inParams);
}
package com.artisan.proxy.dynamic_jdk_proxy;/*** @author 小工匠* @version 1.0* @mark: show me the code , change the world*/
public class RealSubject implements Subject {@Overridepublic void request() {System.out.println("real subject execute request");}@Overridepublic String handle(String inParams) {System.out.println("real subject execute handle, 入参: " + inParams);return inParams;}
}
package com.artisan.proxy.dynamic_jdk_proxy;import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;/*** @author 小工匠* @version 1.0* @mark: show me the code , change the world*/
public class DynamicProxy implements InvocationHandler {private Object target;public DynamicProxy(Object target) {this.target = target;}@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("DynamicProxy before request");Object result = method.invoke(target, args);System.out.println("DynamicProxy after request");return result;}
}
 package com.artisan.proxy.dynamic_jdk_proxy;import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;/*** @author 小工匠* @version 1.0* @mark: show me the code , change the world*/
public class DynamicJdkProxyDemo {public static void main(String[] args) {// 创建真实(返回值请使用接口来接收)Subject subject = new RealSubject();// 创建动态代理类InvocationHandler invocationHandler = new DynamicProxy(subject);// 一切都是面向接口  方式一Subject subjectProxy = (Subject) Proxy.newProxyInstance(Subject.class.getClassLoader(),new Class[]{Subject.class},invocationHandler);// 通过动态代理进行request方法调用subjectProxy.request();System.out.println("------------------------------");String handle = subjectProxy.handle("1-testInParams");System.out.println(handle);System.out.println("------------------------------");// 一切都是面向接口  方式二Subject o = (Subject) Proxy.newProxyInstance(subject.getClass().getClassLoader(),subject.getClass().getInterfaces(),invocationHandler);// 通过动态代理进行request方法调用o.request();System.out.println("------------------------------");String handle1 = o.handle("2-testInParams");System.out.println(handle1);}
}

在这里插入图片描述

通过动态代理,我们在执行类的方法前后成功添加了额外的处理(xxxx),同时客户端代码无需关心具体的日志记录逻辑,实现了解耦。动态代理在很多场景下非常有用,例如AOP(面向切面编程)等


实现方式二 - CGLib 子类代理 (Code Generation Library)

概述

CGLIB(Code Generation Library)是一个开源的第三方库,用于在Java运行时生成字节码并创建代理类。与Java标准库中的动态代理(基于接口)不同,CGLIB代理可以代理普通类,即使它们没有实现任何接口。CGLIB使用ASM库来生成字节码,并通过继承的方式创建代理类,因此也被称为子类代理。CGLIB广泛用于各种框架和库中,如Spring AOP

cglib代理是一种基于字节码技术的代理方式,它可以在运行时动态生成被代理类的子类,并覆盖其中的方法来实现代理。在cglib代理中,被代理类不需要实现任何接口或者继承任何父类。

cglib代理的优点是不需要为每个被代理类编写一个接口或者父类,可以对任意类进行代理,缺点是由于它是基于字节码技术实现的,所以生成的子类可能会比较庞大


pom依赖

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

Code

在这里插入图片描述

 package com.artisan.proxy.dynamic_cglib_proxy;/*** @author 小工匠* @version 1.0* @mark: show me the code , change the world* @desc: 普通类*/
public class NormalClass {public String logic() {System.out.println("Normal class execute Logic");return "OK";}public int minus(int a, int b) {System.out.println("Normal class execute minus");return a - b;}
}
 package com.artisan.proxy.dynamic_cglib_proxy;/*** @author artisan*/
public interface Subject {void request();String handle(String inParams);
}package com.artisan.proxy.dynamic_cglib_proxy;/*** @author 小工匠* @version 1.0* @mark: show me the code , change the world* @desc: 接口类*/
public class RealSubject implements Subject {@Overridepublic void request() {System.out.println("real subject execute request");}@Overridepublic String handle(String inParams) {System.out.println("real subject execute handle, 入参: " + inParams);return inParams;}
}
package com.artisan.proxy.dynamic_cglib_proxy;import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;import java.lang.reflect.Method;/*** @author 小工匠* @version 1.0* @mark: show me the code , change the world* @desc: Cglib动态代理回调类, 需要实现MethodInterceptor接口*/
public class CglibProxy implements MethodInterceptor {/****/private Object target;public CglibProxy(Object target) {this.target = target;}/*** 创建代理类** @return*/public Object createProxy() {// 创建Enhancer对象,用于生成动态代理Enhancer enhancer = new Enhancer();// 设置需要创建代理的类enhancer.setSuperclass(target.getClass());// 设置回调对象,处理代理方法调用enhancer.setCallback(this);// 生成代理类return enhancer.create();}/*** 拦截代理类方法调用** @param o           代理类实例* @param method      代理类方法* @param args        方法参数* @param methodProxy 方法代理,用于调用父类方法*/@Overridepublic Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {// 前置处理System.out.println("CglibProxy before request");// 调用父类方法Object result = methodProxy.invokeSuper(o, args);// 后置处理System.out.println("CglibProxy after request");return result;}
}
package com.artisan.proxy.dynamic_cglib_proxy;/*** @author 小工匠* @version 1.0* @mark: show me the code , change the world*/
public class DynamicCglibProxyDemo {public static void main(String[] args) {testNormalClass();System.out.println("==============================");testInterface();}/*** 代理普通类*/public static void testNormalClass() {// 真实类NormalClass normalClass = new NormalClass();// 构建代理类CglibProxy cglibProxy = new CglibProxy(normalClass);// 创建代理类NormalClass proxy = (NormalClass) cglibProxy.createProxy();// 通过代理类执行方法调用无参方法proxy.logic();System.out.println("---------------------------");// 调用代理对象的方法,传入参数int result = proxy.minus(5, 2);System.out.println(result);}/*** 代理接口*/public static void testInterface() {// 真实类Subject subject = new RealSubject();//  构建代理类CglibProxy cglibProxy = new CglibProxy(subject);// 创建代理类Subject subjectProxy = (Subject) cglibProxy.createProxy();// 通过代理类执行方法调用无参方法subjectProxy.request();System.out.println("---------------------------");// 调用代理对象的方法,传入参数String result = subjectProxy.handle("testInparmas");System.out.println(result);}
}

在这里插入图片描述
通过CGLIB代理,我们成功在方法执行前后添加了额外的处理(xx),实现了解耦。CGLIB代理在不需要接口的情况下也能很好地完成代理任务,但由于它是通过继承的方式生成代理类,可能会影响某些场景,比如无法代理final方法

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

我们常说这个pycharm里有陷阱,第三方库导入失败,看这里!

最近有小伙伴遇到了明明安装了 python 第三方库&#xff0c;但是在 pycharm 当中却导入不成功的问题。 ​ 一直以来&#xff0c;也有不少初学 python 的小伙伴&#xff0c;一不小心就跳进了虚拟环境和系统环境的【陷阱】中。 本文就基于此问题&#xff0c;来说说在 pycharm 当…

PPT颜色又丑又乱怎么办?

一、设计一套PPT时&#xff0c;可以从这5个方面进行设计 二、PPT颜色 &#xff08;一&#xff09;、PPT常用颜色分类 一个ppt需要主色、辅助色、字体色、背景色即可。 &#xff08;二&#xff09;、搭建PPT色彩系统 设计ppt时&#xff0c;根据如下几个步骤&#xff0c;依次选…

基于vue3+webpack5+qiankun实现微前端

一 主应用改造&#xff08;又称基座改造&#xff09; 1 在主应用中安装qiankun(npm i qiankun -S) 2 在src下新建micro-app.js文件&#xff0c;用于存放所有子应用。 const microApps [// 当匹配到activeRule 的时候&#xff0c;请求获取entry资源&#xff0c;渲染到containe…

threejs点击模型实现模型边缘高亮的选中效果--更改后提高帧率

先来个效果图 之前写的那个稍微有点问题&#xff0c;帧率只有30&#xff0c;参照官方代码修改后&#xff0c;帧率可以达到50了&#xff0c;在不全屏的状态下&#xff0c;帧率60 1.首先需要导入库 // 用于模型边缘高亮 import { EffectComposer } from "three/examples/js…

Leetcode-每日一题【剑指 Offer 25. 合并两个排序的链表】

题目 输入两个递增排序的链表&#xff0c;合并这两个链表并使新链表中的节点仍然是递增排序的。 示例1&#xff1a; 输入&#xff1a;1->2->4, 1->3->4输出&#xff1a;1->1->2->3->4->4 限制&#xff1a; 0 < 链表长度 < 1000 解题思路 1…

http协议详解

HTTP是什么&#xff1f; HTTP 是一种用作获取诸如 HTML 文档这类资源的协议。它是 Web 上进行任何数据交换的基础&#xff0c;同时&#xff0c;也是一种客户端—服务器&#xff08;client-server&#xff09;协议&#xff0c;也就是说&#xff0c;请求是由接受方——通常是浏览…

[C++ 网络协议] 套接字和地址族、数据序列

目录 1. 套接字 1.1 在Linux平台下构建套接字 1.1.1 用于接听的套接字(服务器端套接字) 1.1.2 用于发送请求的套接字(客户端套接字) 1.2 在Windows平台下构建套接字 1.2.1 Winsock的初始化 1.2.2 用于接听的套接字(服务器端套接字) 1.2.3 用于发送请求的套接字(客户端套…

Web菜鸟教程-Springboot使用MyBatis生成器生成代码

SpringBoot大大简化了Web开发流程。可以这么说&#xff0c;做Web后来开发大部分时间就是在做配置文件修改。Web开发中&#xff0c;终端的运算能力越来越强&#xff0c;大部分场景就是数据库的操作&#xff0c;只有少部分逻辑会放在Web端处理。而这些增删查改基本属于标准的格式…

Nginx反向代理配置+负载均衡集群部署

文章目录 负载均衡反向代理基础环境部署&#xff1a;什么是代理实验环境图流量过程 环境部署准备两台Web服务器安装Nginx准备页面内容添加主机名 代理服务器配置 修改windos hosts文件测试&#xff1a;终端浏览器 负载均衡反向代理基础环境部署&#xff1a; 什么是代理 正向代…

Python爬虫——selenium的安装和基本使用

1.什么是selenium&#xff1f; selenium是一个用于web应用程序测试的工具selenium测试直接运行在浏览器中&#xff0c;就像真正的用户在操作一样支持通过各种driver&#xff08;FrifoxDriver&#xff0c;ItenrentExploreDriver&#xff0c;OperaDriver&#xff0c;ChromeDrive…

Android模板设计模式之 - 构建整个应用的BaseActivity

1. 模式介绍 模式的定义 定义一个操作中的算法的框架&#xff0c;而将一些步骤延迟到子类中。使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤。 模式的使用场景 1.多个子类有公有的方法&#xff0c;并且逻辑基本相同时。 2.重要、复杂的算法&#xff0c;可…

threejs中gltf模型出现的问题(黑色,颜色不协调,太小)和解决方案

模型一片漆黑 如下图 可能原因&#xff0c;没有灯光&#xff0c;加下以下代码&#xff1a; // 4、加入灯光 const lightness new THREE.HemisphereLight(0xffffff, 0x444444); lightness.position.set(0, 20, 0); scene.add(lightness); const shadowLight new THREE.Direct…

实现同时查找多个关键词——KeywordCrafter - 关键词匠心

具体功能&#xff1a;同时查找多个关键词&#xff0c;高亮加粗显示&#xff0c;并关键词显示出现次数。 &#x1f9d0;碎碎念&#xff1a;最近在写文案的时候&#xff0c;总是要避免出现一个敏感词汇&#xff0c;利用 (commandF) or (CtrF) 查找&#xff0c;只能一个一个单词去…

中国艺术孙溟㠭篆刻作品《得大自在》

关汉卿《四块玉闲适》&#xff1a;“适意行&#xff0c;安心坐。渴时饮&#xff0c;饥时餐&#xff0c;醉时歌。困来时就向莎茵卧。日月长&#xff0c;天地阔&#xff0c;闲快活。” 整理/释门

JS逆向系列之猿人学爬虫第14题-备而后动-勿使有变

文章目录 题目地址参数分析参考jspython 调用往期逆向文章推荐题目地址 https://match.yuanrenxue.cn/match/14题目难度标的是困难,主要难在js混淆部分。 参数分析 初始抓包有无限debugger反调试,可以直接hook 函数构造器过掉无限debugger Function.prototype.__construc…

[FPAG开发]使用Vivado创建第一个程序

1 打开Vivado软件&#xff0c;新建项目 选择一个纯英文路径 选择合适的型号 产品型号ZYNQ-7010xc7z010clg400-1ZYNQ-7020xc7z010clg400-2 如果型号选错&#xff0c;可以单击这里重新选择 2 创建工程源文件 可以看到文件创建成功 双击文件打开&#xff0c;插入代码 modul…

IOC容器

DI&#xff08;依赖注入&#xff09;&#xff1a;DI&#xff08;Dependency Injection&#xff09;是一种实现松耦合和可测试性的软件设计模式。它的核心思想是将依赖关系的创建与管理交给外部容器&#xff0c;使得对象之间只依赖于接口而不直接依赖于具体实现类。通过依赖注入…

【分布式存储】数据存储和检索~B+树

为什么数据存储结构重要 在存储系统中&#xff0c;其实不管数据是什么样的&#xff0c;归根结底其实都还是取决于数据的底层存储结构&#xff0c;而主要常见的就是数据库索引结构&#xff0c;B树、Redis中跳表、以及LSM、搜索引擎中的倒排索引。本质都是如何利用不用的数据结构…

Linux下grep通配容易混淆的地方

先上一张图: 我希望找到某个版本为8的一个libXXX.8XXX.so ,那么应该怎么写呢? 先看这种写法对不对: 是不是结果出乎你的意料之外? 那么我们来看一下规则: 这里的 "*" 表示匹配前一个字符的零个或多个 于是我们就不难理解了: lib*8*.so 表示 包…