第 15 章 代理模式

1、代理模式的基本介绍

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

2、静态代码模式

静态代码模式的基本介绍

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

静态代理应用实例

具体要求

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

类图


代码实现

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

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

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

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

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

静态代理优缺点

  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)
    

动态代理应用实例

应用实例要求

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


类图


代码实现

  1. ITeacherDao

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

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

    public class ProxyFactory {// 维护一个目标对象 , Objectprivate 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;}});}}
    
  4. Client

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

代码追踪

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

  2. 参数 proxy 为代理对象

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

    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 implementationprivate String              name;private Class<?>            returnType;private Class<?>[]          parameterTypes;private Class<?>[]          exceptionTypes;private int                 modifiers;// Generics and annotations supportprivate transient String              signature;// generic info repository; lazily initializedprivate transient MethodRepository genericInfo;private byte[]              annotations;private byte[]              parameterAnnotations;private byte[]              annotationDefault;private volatile MethodAccessor methodAccessor;
    
  4. 参数 args 为方法的形参

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文件

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

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

Cglib代理模式应用实例

应用实例要求

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



代码实现

  1. TeacherDao:被代理类

    public class TeacherDao {public String teach() {System.out.println(" 老师授课中  , 我是cglib代理,不需要实现接口 ");return "hello";}}
    
  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 方法,会调用目标对象的方法@Overridepublic 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;}}
    
  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);}}
    

代码追踪

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

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

    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 implementationprivate String              name;private Class<?>            returnType;private Class<?>[]          parameterTypes;private Class<?>[]          exceptionTypes;private int                 modifiers;// Generics and annotations supportprivate transient String              signature;// generic info repository; lazily initializedprivate transient MethodRepository genericInfo;private byte[]              annotations;private byte[]              parameterAnnotations;private byte[]              annotationDefault;private volatile MethodAccessor methodAccessor;
    

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

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

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

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

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

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

第 15 章 代理模式相关推荐

  1. 【设计模式】第15章·代理模式

    一.代理模式概述 由于某些原因,客户端不想或不能直接访问一个对象,此时可以通过一个称为"代理"的第三者来实现间接访问,该方案对应的设计模式被称为代理模式. 定义如下: 代理模式:给 ...

  2. 【设计模式】第二章 代理模式

    第二章 代理模式 文章目录 第二章 代理模式 一.简介 一.静态代理 二.动态代理 1.JDK 代理 2.CGLib 代理 三.总结 一.简介 你可能会对这个设计模式感到陌生,但是你对 Spring ...

  3. 设计模式笔记15:代理模式(Proxy Pattern)

    一.代理模式的内容        代理模式为另一个对象提供一个替身或占位符以访问这个对象.         给某一个对象提供一个代理,并由代理对象控制对原对象的引用.代理模式的英文叫做Proxy或Su ...

  4. Head First设计模式读书笔记十 第十一章 代理模式

    之前的总结链接 https://blog.csdn.net/u011109881/article/details/59191308 代理模式本身的框架结构并不难理解,但是在实际应用中,我们往往被实际应 ...

  5. 设计模式完结(12)-- 代理模式

    代理模式与装饰模式区别: 装饰器模式关注于在一个对象上动态的添加方法,然而代理模式关注于控制对对象的访问.换句话说,用代理模式,代理类(proxy class)可以对它的客户隐藏一个对象的具体信息. ...

  6. C#设计模式之代理模式(一)

    代理模式是常用的结构型设计模式之一,当无法直接访问某个对象或访问某个对象存在困难时可以通过一个代理对象来间接访问,为了保证客户端使用的透明性,所访问的真实对象与代理对象需要实现相同的接口.根据代理模式 ...

  7. 代理模式(一):代理模式概述,代理模式结构与实现

    代理模式是常用的结构型设计模式之一,当无法直接访问某个对象或访问某个对象存在困难时可以通过一个代理对象来间接访问,为了保证客户端使用的透明性,所访问的真实对象与代理对象需要实现相同的接口.根据代理模式 ...

  8. 代理模式(四):代理模式效果与适用场景

    15.7 代理模式效果与适用场景 代理模式是常用的结构型设计模式之一,它为对象的间接访问提供了一个解决方案,可以对对象的访问进行控制.代理模式类型较多,其中远程代理.虚拟代理.保护代理等在软件开发中应 ...

  9. 第四章 Caché 设计模式 代理模式

    文章目录 第四章 Caché 设计模式 代理模式 定义 类型 使用场景 优点 缺点 结构图 完整示例 抽象主题类 真实主题类 代理类 对象类 调用 思考 第四章 Caché 设计模式 代理模式 定义 ...

最新文章

  1. 超市的100年发展史:好日子终结 亚马逊无人店是趋势
  2. JavaScript之三:jQuery插件开发(一)
  3. MySQL的介绍以及使用
  4. 内置对象—request
  5. 浅谈python_浅谈python-Django
  6. python函数:函数参数,常用函数工具
  7. SSM项目使用example查询时多次查询条件相同
  8. 20181225 基于TCP/IP和基于UDP/IP的套接字编程
  9. 20190703 日子
  10. 转载--Ajax学习---DOM进行Web响应
  11. Java Restful风格-Jersey RESTful 框架入门
  12. 《大师谈游戏设计——创意与节奏》【笔记一】
  13. 智能陈桥五笔输入法 for linux,解决在Linux下安装陈桥五笔输入法的方法
  14. “5G+”发展论坛暨“金帽子”年度盛典圆满结束,共同探讨5G背景下网安技术发展和前沿趋势
  15. PS cc 2018安装教程
  16. 嵌入式分享合集118
  17. 集成微软更新汇总KB3125574
  18. python基础知识集(三)
  19. 拍照翻译的软件有什么?不妨试试这几款好用的软件
  20. unity网络实战开发(丛林战争)-正式开发阶段(018-声音管理器模块的完善)

热门文章

  1. 2021年中国乙酸异冰片酯市场趋势报告、技术动态创新及2027年市场预测
  2. 下拉样式_Axure 组件重写之神奇的文本框和下拉框
  3. 【IT】使用gdb调试code
  4. 昇腾 AI 成就了一群玩船模的大学生——创新,有“模”有 Young
  5. “再见,Linux!”
  6. 字节跳动异构场景下的高可用建设实践
  7. 重大活动网络攻击面前,京东智联云的攻防之道
  8. 编程基础知识真的是九阳神功吗?
  9. 2020 年,你必须知道发展最快和衰落最快的软件技能
  10. “瘟疫”笼罩下的物联网危与机