设计模式—代理模式

news2025/1/19 18:16:36

文章目录

  • 以前自己做的笔记
      • 动态代理(重点)
      • 1.基于jdk的动态代理
      • 2.基于cglib的动态代理
  • 新资料
    • 第 15 章 代理模式
      • 1、代理模式的基本介绍
      • 2、静态代码模式
      • 3、动态代理模式
      • 4、Cglib 代理模式
      • 5、代理模式(Proxy)的变体

代理模式是给某一个对象提供一个代理,并通过代理对象访问目标对象。从而实现在不修改原有类代码的情况下,为原有类添加额外的功能。

代理模式有不同的形式,主要有三种:静态代理、 动态代理 (JDK代理、接口代理)和 Cglib代理 (可以在内存动态的创建对象,而不需要实现接口, 他是属于动态代理的范畴) 。

以前自己做的笔记

动态代理(重点)

		* 增强对象的功能:
			 	* 实现方式:
				 	1. 静态代理:有一个类文件描述代理模式
				 	2. 动态代理:在内存中形成代理类
						* 动态代理实现步骤:
							1. 代理对象和真实对象实现相同的接口
							2. 代理对象 = Proxy.newProxyInstance();
							3. 使用代理对象调用方法。
							4. 增强方法

						* 增强方式:
							1. 增强参数列表
							2. 增强返回值类型
							3. 增强方法体执行逻辑	

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

代理对象如何增强真实对象的功能?

* 增强方式:
	1. 增强参数列表
	2. 增强返回值类型
	3. 增强方法体执行逻辑	

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

1.基于jdk的动态代理

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

2.基于cglib的动态代理

代码

在这里插入图片描述
在这里插入图片描述

新资料

第 15 章 代理模式

1、代理模式的基本介绍

  1. 代理模式:为一个对象提供一个替身,以控制对这个对象的访问。 即通过代理对象访问目标对象。
  2. 代理对象的好处是:可以在目标对象实现的基础上,增强额外的功能操作,即扩展目标对象的功能。
  3. 被代理的对象可以是远程对象、 创建开销大的对象或需要安全控制的对象
  4. 代理模式有不同的形式,主要有三种:静态代理、 动态代理 (JDK代理、接口代理)和 Cglib代理 (可以在内存动态的创建对象,而不需要实现接口, 他是属于动态代理的范畴) 。

image-20200827170912500

2、静态代码模式

静态代码模式的基本介绍

静态代理在使用时,需要定义接口或者父类,被代理对象(即目标对象)与代理对象一起实现相同的接口或者是继承相同父类

静态代理应用实例

具体要求

  1. 定义一个接口:ITeacherDao
  2. 目标对象TeacherDAO实现接口ITeacherDAO
  3. 使用静态代理方式,就需要在代理对象TeacherDAOProxy中也实现ITeacherDAO
  4. 调用的时候通过调用代理对象的方法来调用目标对象
  5. 特别提醒:代理对象与目标对象要实现相同的接口,然后通过调用相同的方法来调用目标对象的方法

类图

image-20200827173045461


代码实现

  1. ITeacherDao:定义抽象的方法(授课)规范

    //接口
    public interface ITeacherDao {
    
    	void teach(); // 授课的方法
    	
    }
    123456
    
  2. TeacherDao:实现具体的业务功能

    public class TeacherDao implements ITeacherDao {
    
    	@Override
    	public void teach() {
    		System.out.println(" 老师授课中  。。。。。");
    	}
    
    }
    12345678
    
  3. TeacherDaoProxy:代理类,聚合了一个 ITeacherDao 具体实现类的对象,在TeacherDaoProxy实现了 ITeacherDao 接口,并在 teach() 方法中完成代理操作

    //代理对象,静态代理
    public class TeacherDaoProxy implements ITeacherDao {
    
    	private ITeacherDao target; // 目标对象,通过接口来聚合
    
    	// 构造器
    	public TeacherDaoProxy(ITeacherDao target) {
    		this.target = target;
    	}
    
    	@Override
    	public void teach() {
    		System.out.println("开始代理  完成某些操作。。。。。 ");// 方法
    		target.teach();
    		System.out.println("提交。。。。。");// 方法
    	}
    
    }
    123456789101112131415161718
    
  4. Client:客户端

    public class Client {
    
    	public static void main(String[] args) {
    		// 创建目标对象(被代理对象)
    		TeacherDao teacherDao = new TeacherDao();
    
    		// 创建代理对象, 同时将被代理对象传递给代理对象
    		TeacherDaoProxy teacherDaoProxy = new TeacherDaoProxy(teacherDao);
    
    		// 通过代理对象,调用到被代理对象的方法
    		// 即:执行的是代理对象的方法,代理对象再去调用目标对象的方法
    		teacherDaoProxy.teach();
    	}
    
    }
    123456789101112131415
    

静态代理优缺点

  1. 优点:在不修改目标对象的功能前提下,能通过代理对象对目标功能扩展
  2. 缺点:因为代理对象需要与目标对象实现一样的接口,所以会有很多代理类
  3. 一旦接口增加方法,目标对象与代理对象都要维护

3、动态代理模式

动态代理模式的基本介绍

  1. 代理对象不需要实现接口, 但是目标对象要实现接口, 否则不能用动态代理
  2. 代理对象的生成, 是利用JDKAPI,动态的在内存中构建代理对象
  3. 动态代理也叫做: JDK代理、 接口代理

JDK中生成代理对象的API

  1. 代理类所在包:java.lang.reflect.Proxy

  2. JDK实现代理只需要使用newProxyInstance方法,但是该方法需要接收三个参数,完整的写法是:

    static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)
    1
    

动态代理应用实例

应用实例要求

将前面的静态代理改进成动态代理模式(即:JDK代理模式)


类图

image-20200827184724665


代码实现

  1. ITeacherDao

    //接口
    public interface ITeacherDao {
    
    	void teach(); // 授课方法
    
    	void sayHello(String name);
    	
    }
    12345678
    
  2. TeacherDao

    public class TeacherDao implements ITeacherDao {
    
    	@Override
    	public void teach() {
    		System.out.println(" 老师授课中.... ");
    	}
    
    	@Override
    	public void sayHello(String name) {
    		System.out.println("hello " + name);
    	}
    
    }
    12345678910111213
    
  3. ProxyFactory:通过构造器传入被代理对象,通过 Proxy.newProxyInstance() 方法中的 new InvocationHandler() 匿名内部类实现具体的代理逻辑

    public class ProxyFactory {
    
    	// 维护一个目标对象 , Object
    	private Object target;
    
    	// 构造器 , 对target 进行初始化
    	public ProxyFactory(Object target) {
    		this.target = target;
    	}
    
    	// 给目标对象 生成一个代理对象
    	public Object getProxyInstance() {
    
    		// 说明
    		/*
    		 * public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)
    		 * 
    		 * 1. ClassLoader loader : 指定当前目标对象使用的类加载器, 获取加载器的方法固定 
    		 * 2. Class<?>[] interfaces: 目标对象实现的接口类型,使用泛型方法确认类型 
    		 * 3. InvocationHandler h : 事情处理,执行目标对象的方法时,会触发事情处理器方法, 会把当前执行的目标对象方法作为参数传入
    		 */
    		return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(),
    				new InvocationHandler() {
    					@Override
    					/*
    					 * proxy:proxy the proxy instance that the method was invoked on
    					 * method:待调用的目标方法(target 的方法)
    					 * args:方法参数
    					 */
    					public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    						System.out.println("JDK代理开始~~");
    						// 反射机制调用目标对象的方法
    						Object returnVal = method.invoke(target, args);
    						System.out.println("JDK代理提交");
    						return returnVal;
    					}
    				});
    	}
    
    }
    12345678910111213141516171819202122232425262728293031323334353637383940
    
  4. Client

    public class Client {
    
    	public static void main(String[] args) {
    		// 创建目标对象
    		ITeacherDao target = new TeacherDao();
    
    		// 给目标对象,创建代理对象, 可以转成 ITeacherDao
    		ITeacherDao proxyInstance = (ITeacherDao) new ProxyFactory(target).getProxyInstance();
    
    		// proxyInstance=class com.sun.proxy.$Proxy0 内存中动态生成了代理对象
    		System.out.println("proxyInstance=" + proxyInstance.getClass());
    
    		// 通过代理对象,调用目标对象的方法
    		proxyInstance.teach();
    
    		proxyInstance.sayHello(" tom ");
    	}
    
    }
    12345678910111213141516171819
    

代码追踪

  1. Step into 进入 proxyInstance.sayHello(" tom ");

    image-20200827185040357

  2. 参数 proxy 为代理对象

    image-20200827185342189

  3. 参数 method 为被代理(target)对象中的方法

    image-20200827185415158

    public final class Method extends Executable {
        private Class<?>            clazz;
        private int                 slot;
        // This is guaranteed to be interned by the VM in the 1.4
        // reflection implementation
        private String              name;
        private Class<?>            returnType;
        private Class<?>[]          parameterTypes;
        private Class<?>[]          exceptionTypes;
        private int                 modifiers;
        // Generics and annotations support
        private transient String              signature;
        // generic info repository; lazily initialized
        private transient MethodRepository genericInfo;
        private byte[]              annotations;
        private byte[]              parameterAnnotations;
        private byte[]              annotationDefault;
        private volatile MethodAccessor methodAccessor;
    123456789101112131415161718
    
  4. 参数 args 为方法的形参

    image-20200827190544919

4、Cglib 代理模式

Cglib代理模式的基本介绍

  1. 静态代理和JDK代理模式都要求目标对象是实现一个接口,但是有时候目标对象只是一个单独的对象,并没有实现任何的接口,这个时候可使用目标对象子类来实现代理,这就是Cglib代理

  2. Cglib代理也叫作子类代理,它是在内存中构建一个子类对象从而实现对目标对象功能扩展,有些书也将Cglib代理归属到动态代理

  3. Cglib是一个强大的高性能的代码生成包,它可以在运行期扩展java类与实现java接口,它广泛的被许多AOP的框架使用,例如Spring AOP,实现方法拦截

  4. AOP
    

    编程中如何选择代理模式:

    1. 目标对象需要实现接口,用JDK代理
    2. 目标对象不需要实现接口,用Cglib代理
  5. Cglib包的底层是通过使用字节码处理框架ASM来转换字节码并生成新的类

Cglib代理模式实现步骤

  1. 需要引入cglibjar文件

    image-20200827212642592

  2. 在内存中动态构建子类, 注意代理的类不能为final,否则报错java.lang.IllegalArgumentException

  3. 目标对象的方法如果为finalstatic,那么就不会被拦截,即不会执行目标对象额外的业务方法

Cglib代理模式应用实例

应用实例要求

将前面的案例用Cglib代理模式实现


image-20200827230207986


代码实现

  1. TeacherDao:被代理类

    public class TeacherDao {
    
    	public String teach() {
    		System.out.println(" 老师授课中  , 我是cglib代理,不需要实现接口 ");
    		return "hello";
    	}
    	
    }
    12345678
    
  2. ProxyFactory:代理工厂类,用于获取代理对象

    public class ProxyFactory implements MethodInterceptor {
    
    	// 维护一个目标对象
    	private Object target;
    
    	// 构造器,传入一个被代理的对象
    	public ProxyFactory(Object target) {
    		this.target = target;
    	}
    
    	// 返回一个代理对象: 是 target 对象的代理对象
    	public Object getProxyInstance() {
    		// 1. 创建一个工具类
    		Enhancer enhancer = new Enhancer();
    		// 2. 设置父类
    		enhancer.setSuperclass(target.getClass());
    		// 3. 设置回调函数
    		enhancer.setCallback(this);
    		// 4. 创建子类对象,即代理对象
    		return enhancer.create();
    	}
    
    	// 重写 intercept 方法,会调用目标对象的方法
    	@Override
    	public Object intercept(Object arg0, Method method, Object[] args, MethodProxy arg3) throws Throwable {
    		System.out.println("Cglib代理模式 ~~ 开始");
    		Object returnVal = method.invoke(target, args);
    		System.out.println("Cglib代理模式 ~~ 提交");
    		return returnVal;
    	}
    
    }
    1234567891011121314151617181920212223242526272829303132
    
  3. Client:客户端

    public class Client {
    
    	public static void main(String[] args) {
    		// 创建目标对象
    		TeacherDao target = new TeacherDao();
    		
    		// 获取到代理对象,并且将目标对象传递给代理对象
    		TeacherDao proxyInstance = (TeacherDao) new ProxyFactory(target).getProxyInstance();
    
    		// 执行代理对象的方法,触发intecept 方法,从而实现 对目标对象的调用
    		String res = proxyInstance.teach();
    		System.out.println("res=" + res);
    	}
    
    }
    123456789101112131415
    

代码追踪

  1. Step into 进入 String res = proxyInstance.teach();

    image-20200827232343715

  2. 形参 Methodpublic java.lang.String com.atguigu.proxy.cglib.TeacherDao.teach()

    image-20200827232144951

    public final class Method extends Executable {
        private Class<?>            clazz;
        private int                 slot;
        // This is guaranteed to be interned by the VM in the 1.4
        // reflection implementation
        private String              name;
        private Class<?>            returnType;
        private Class<?>[]          parameterTypes;
        private Class<?>[]          exceptionTypes;
        private int                 modifiers;
        // Generics and annotations support
        private transient String              signature;
        // generic info repository; lazily initialized
        private transient MethodRepository genericInfo;
        private byte[]              annotations;
        private byte[]              parameterAnnotations;
        private byte[]              annotationDefault;
        private volatile MethodAccessor methodAccessor;
    123456789101112131415161718
    

5、代理模式(Proxy)的变体

几种常见的代理模式介绍——几种变体

  1. 防火墙代理:内网通过代理穿透防火墙,实现对公网的访问

  2. 缓存代理:比如:当请求图片文件等资源时,先到缓存代理去取,如果取到资源则ok,如果取不到资源,再到公网或者数据库取,然后缓存

  3. 远程代理远程对象的本地代表,通过它可以把远程对象当本地对象来调用。远程代理通过网络和真正的远程对象沟通信息

    image-20200827232506561

  4. 同步代理: 主要使用在多线程编程中,完成多线程间同步工作

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

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

相关文章

第12章 网络 (6)

12.8 网络层 12.8.4 分组转发 转发IP分组&#xff0c;根据目标地址分为&#xff1a; 1. 直接和本地相连。 2. 不直接相连&#xff0c;需要网关转发。 int ip_route_input_noref(skb, daddr, saddr, tos, net_dev)&#xff1a; //查找路由表。 如果 skb->_skb_r…

安捷伦色谱仪器LabVIEW软件替换与禁运配件开发

可行性分析及实现路径 可行性&#xff1a; 软件替换&#xff1a; 驱动程序支持&#xff1a; 要实现LabVIEW对安捷伦色谱仪器的控制&#xff0c;需要检查安捷伦是否提供LabVIEW驱动程序。如果没有现成的驱动&#xff0c;则可能需要开发自定义的驱动程序&#xff0c;通过LabVIEW…

微软推出全新多语言高质量Phi-3.5语言模型

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

css flex布局 justify-content: space-between 最后两张居左

比如如果是8张&#xff0c;最后两张两边对齐&#xff0c;第八张最后一张 放个占位符就OK了 <div class"previewPadding flex" > <div class"picList picList3" v-for"(item,index) in picDataList" :key"index"> <…

6个免费字体网站,无需担心版权问题~

在设计项目中&#xff0c;选择合适的字体至关重要。然而&#xff0c;许多高质量的字体往往价格不菲。幸运的是&#xff0c;有一些网站提供了免费的商用字体&#xff0c;既能满足设计需求&#xff0c;又不需要额外的预算。在这篇文章中&#xff0c;分享6个免费商用字体网站&…

济南网站制作方案定制

在当今数字化时代&#xff0c;拥有一个专业的网站已经成为企业发展不可或缺的一部分。济南作为山东省的省会&#xff0c;经济发展迅速&#xff0c;各行各业对网站制作的需求也日益增加。因此&#xff0c;定制化的网站制作方案在济南显得尤为重要&#xff0c;能够帮助企业在激烈…

深入探究为什么 RAG 并不总是按预期工作:概述其背后的业务价值、数据和技术。

添加图片注释&#xff0c;不超过 140 字&#xff08;可选&#xff09; 欢迎来到雲闪世界。我们将首先探讨决定基于 RAG 的项目成败的业务要素。然后&#xff0c;我们将深入探讨常见的技术障碍&#xff08;从数据处理到性能优化&#xff09;&#xff0c;并讨论克服这些障碍的策略…

数据结构(邓俊辉)学习笔记】优先级队列 10——左式堆:插入 + 删除

文章目录 1. 插入即是合并2. 删除亦是合并 1. 插入即是合并 以上&#xff0c;我们已经实现了&#xff0c;对于左式堆来说最为在意的合并算法。非常有意思的是&#xff0c;尽管合并操作并非优先级队列所要求的基本操作接口。但基于合并操作&#xff0c;我们却同样可以实现左式堆…

超全大模型训练流程,教你如何训练自己的大模型

“大模型的核心主要有两部分&#xff0c;一是训练数据&#xff0c;二是机器学习模型。” 现在大模型发展得如火如荼&#xff0c;但是没有学过人工智能技术的开发者&#xff0c;只会调用其接口&#xff0c;但不清楚怎么训练一个大模型。 今天就简单介绍一下自己的理解&#xf…

Transformer系列-10丨一文理解透Transformer

一、引言 "Attention Is All You Need"是一篇于2017年发表的开创性论文&#xff0c;首次介绍了Transformer模型。 这篇论文彻底改变了自然语言处理&#xff08;NLP&#xff09;领域的研究方向&#xff0c;为后续的众多NLP模型和应用奠定了基础。我们熟知的ChatGPT也…

【022】字符串的处理(输出,分割,删除,新增,替换,查找,长度)_#VBA

字符串的处理——输出,分割,删除,新增,替换,查找,长度 字符串的处理1. 输出2. 长度3. 查找4. 删除5. 新增6. 分割7. 替换字符串的处理 为了更好快捷查找对应的字符串处理方法,将对应的方法汇总,可以直接使用,没有过多的介绍,直接代码块及对应效果。包括字符串的输出…

全国上市公司网络安全风险指数(2001-2023年)

数据来源&#xff1a;本数据参考耿勇老师等&#xff08;2024&#xff09;做法采集了2001-2023年的上市公司年报&#xff0c;所有年报均来自于深交所和上交所官方网站&#xff0c;通过对上市公司的年报进行精读&#xff0c;提取出包括网络安全、网络攻击等在内的39个关键词构成企…

自定义@ResponseBody以及SpringMVC总结

文章目录 1.需求分析2.目录3.自定义ResponseBody注解4.MonsterController.java5.Monster.java 实现序列化接口6.引入jackson7.Adapter.java 如果有ResponseBody注解就返回json8.测试9.SpringMVC执行流程 1.需求分析 2.目录 3.自定义ResponseBody注解 package com.sunxiansheng…

大数据技术之 Flume概述、安装(1)

目录 Flume 概述 Flume 定义 为什么选用 Flume Flume 基础架构 Agent Source Sink Channel Event Flume 安装 Flume 安装部署 安装地址 安装部署 Flume 概述 Flume 定义 Flume 是 Cloudera 提供的一个高可用的、高可靠的、分布式的海量日志采集、聚合和传输的系统。Flume…

多系统萎缩不慌张,这些维生素是你的“守护神”✨

亲爱的朋友们&#xff0c;今天我们来聊聊一个可能不太为人熟知但至关重要的健康话题——多系统萎缩&#xff08;MSA&#xff09;。面对这样的挑战&#xff0c;除了医疗治疗&#xff0c;日常的营养补充也是不可或缺的一环。特别是维生素&#xff0c;它们在我们的身体中扮演着举足…

有无符号整形加减,截断,提升等问题解析

一&#xff1a;整形截断问题 1. 分析&#xff1a;-128 的原码是 10000000 00000000 00000000 10000000 补码是 11111111 11111111 11111111 10000000&#xff0c;因为是char 型&#xff0c;截断之后&#xff1a;10000000 %u 是打印无符号整数&#xff0c;整形提升补截断之后…

多态(详细介绍以及内存图展示)

什么是多态&#xff1f; 同类型的对象&#xff0c;表现出的不同形态 多态的表现形式 父类类型 对象名称 子类对象 多态的前提 1.有继承关系 2.有父类引用指向子类 Fu f new Zi() 3.有方法重写 多态的好处 使用父类型作为参数&#xff0c;可以接收所有子类对象 体现…

大模型从入门到精通,看这篇就够了,AI小白的大模型学习路径_大模型教程

写这篇文章的初衷&#xff1a;作为一个AI小白&#xff0c;把我自己学习大模型的学习路径还原出来&#xff0c;包括理解的逻辑、看到的比较好的学习材料&#xff0c;通过一篇文章给串起来&#xff0c;对大模型建立起一个相对体系化的认知&#xff0c;才能够在扑面而来的大模型时…

牛客小白月赛99

文章目录 A.材料打印B. %%%C.迷宫又是一年毕业季题目链接 A.材料打印 签到题&#xff0c;直接按照题意输出就行。赛时写的有点慢了&#xff0c;这种题应该一分钟之内写完的 void solve () {int n;cin>>n;for (int i1;i<n;i) {int a,b,c,d;cin>>a>>b>…

go const(常量)

常量介绍 示例 package mainimport ("fmt" )func main() {const name "tom"fmt.Println(name)const tax float64 0.8fmt.Println(tax) }go run const.go tom 0.8package mainimport ("fmt" )func main() {const a intfmt.Println(a) }go run…