设计模式-------------静态/动态代理模式(结构型设计模式)

简介: 本文详细介绍了代理模式,包括其定义、应用场景、UML类图、代码实现和实际例子,阐述了静态代理和动态代理的区别以及它们的优缺点,展示了如何通过代理模式来控制对目标对象的访问并增强其功能。

文章目录

  • 1、代理模式定义
  • 2、应用场景
  • 3、UML类图
  • 4、通用代码实现
  • 5、结果
  • 6、实际例子(静态代理)
  • 7、动态代理生成对象的步骤
  • 8、代理模式的优缺点
  • 9、静态代理和动态代理的区别

1、代理模式定义

为其他对象提供一种代理,控制对对这个对象的访问。(你去找租房子、不是直接找房东。而是通过中介所给你提供房东的信息、然后你再去找对应的房东)

2、应用场景

  • 租房中介
  • 婚姻介绍
  • 中介找工作(打工人通过中介去电子厂打螺丝)

3、UML类图

在这里插入图片描述

  • 抽象主题角色:声明真实主题和代理主题的共同接口方法
  • 真实主题角色:被代理类,代表的真实对象,负责执行系统的真正的逻辑业务对象
  • 代理主题角色:代理类,持有RealSubject的引用,完全具备对RealSubject的代理权

(代理中持有子类实现的父类引用,就可以调用子类的真实实现方法,而且还可以在调用该方法之前。做一些额外的事情。代码功能增强、和spirng中的aop切面有点象。比如你要结婚,婚礼的布置和婚后的收拾,交给代理类实现。你只需要去结婚就行。这个婚礼的布置和收拾就像代码功能增强)

4、通用代码实现

package com.zheng.demo4;

public class Client {

    public static void main(String[] args) {

        new Proxy(new RealSubject()).request();

    }


    //抽象主题角色
    interface ISubject {
        public void request();//要做的事情
    }

    //真实角色
    static public class RealSubject implements ISubject {

        @Override
        public void request() {
            System.out.println("我要结婚啦");
        }
    }


    //代理角色
    static class Proxy implements ISubject {
        ISubject subject;

        public Proxy(ISubject subject) {
            this.subject = subject;
        }

        @Override
        public void request() {
            before();
            subject.request();//调用真实对象的业务逻辑
            after();
        }

        //方法增强、结婚前干的事情
        public void before() {
            System.out.println("布置婚礼现场");
        }

        //结婚后干的事情
        public void after() {
            System.out.println("婚礼现场的清理");
        }

    }
}

5、结果

在这里插入图片描述

6、实际例子(静态代理)

年轻人被催婚、介绍对象
代码结构
在这里插入图片描述
一个是静态代理、一个是动态代理。有对静态代理的改进为动态代理。
抽象

package com.zheng.demo4;

public interface IPerson {
    void findLove();
}

具体的人

package com.zheng.demo4;

public class LiSi implements IPerson {
    @Override
    public void findLove() {
        System.out.println("李四提出结婚对象的要求");
    }
}

代理

package com.zheng.demo4;


public class LiSiFather implements IPerson {
    LiSi liSi;

    public LiSiFather(LiSi liSi) {
        this.liSi = liSi;
    }

    @Override
    public void findLove() {
        liSi.findLove();
        System.out.println("李四的老爹开始寻找");
        System.out.println("试着交往");

    }

}

客户端

package com.zheng.demo4;

public class TestClient {
    public static void main(String[] args) {
        new  PersonProxy(new LiSi()).findLove();
    }
}

结果
在这里插入图片描述

缺点:李四的老爹只能给李四找对象、不能给其他人找对象。

解决方法、使用动态代理、把代理类搞成一个中介、给谁都可以介绍。动态代理的底层一般不用实现,直接调用现成的API。

package com.zheng.demo4;

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

public class MeiPoProxy implements InvocationHandler {

    private IPerson target;


    public IPerson getInstance(IPerson target) {
        this.target = target;
        Class<?> clazz = target.getClass();
        return (IPerson) Proxy.newProxyInstance(clazz.getClassLoader(), clazz.getInterfaces(), this);
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        before();
        Object result = method.invoke(this.target, args);
        after();
        return result;
    }

    //方法增强
    public void before() {
        System.out.println("我是婚姻中介所、收到了你的对象要求");
    }

    //方法增强
    public void after() {
        System.out.println("交换资料、开始交往吧");
    }
}

客户端调用

package com.zheng.demo4;

public class TestClient {
    public static void main(String[] args) {
        new MeiPoProxy().getInstance(new LiSi()).findLove();
    }
}

结果:
在这里插入图片描述

调用代理类的newProxyInstance方法
在这里插入图片描述
进入这个方法可以看到生成一个实例对象
在这里插入图片描述

在这里插入图片描述

7、动态代理生成对象的步骤

  • 1、获取被代理对象的引用,并且获取它的所有接口,反射获取
  • 2、JDK动态代理类重新生成一个新的类,同时新的类要实现被代理类实现的所有接口
  • 3、动态生成java代码,新加的业务逻辑方法由一定的逻辑代码调用
  • 4、编译新生成的java代码.class 文件
  • 5、重新加载到JVM中运行。

8、代理模式的优缺点

优点:

  • 代理对象和真实对象分开
  • 降低系统耦合性,扩展性好
  • 保护目标对象
  • 增强目标对象功能

缺点:

  • 类的数量增加
  • 增加代理对象,导致处理请求变慢
  • 增加系统复杂度

9、静态代理和动态代理的区别

  • 静态代理只能通过手动完成,被代理类增加新的方法,代理类也需要同步增加,违背开闭原则
  • 动态代理采用在运行时动态生成代码,取消了对被代理类的扩展限制,遵循开闭原则
  • 动态代理对目标类的增强逻辑进行扩展,结合策略模式
相关文章
|
5天前
|
设计模式 算法 安全
设计模式——模板模式
模板方法模式、钩子方法、Spring源码AbstractApplicationContext类用到的模板方法
设计模式——模板模式
|
1月前
|
设计模式
设计模式-单一职责模式
设计模式-单一职责模式
|
1月前
|
设计模式 XML 存储
【二】设计模式~~~创建型模式~~~工厂方法模式(Java)
文章详细介绍了工厂方法模式(Factory Method Pattern),这是一种创建型设计模式,用于将对象的创建过程委托给多个工厂子类中的某一个,以实现对象创建的封装和扩展性。文章通过日志记录器的实例,展示了工厂方法模式的结构、角色、时序图、代码实现、优点、缺点以及适用环境,并探讨了如何通过配置文件和Java反射机制实现工厂的动态创建。
【二】设计模式~~~创建型模式~~~工厂方法模式(Java)
|
1月前
|
设计模式 XML Java
【一】设计模式~~~创建型模式~~~简单工厂模式(Java)
文章详细介绍了简单工厂模式(Simple Factory Pattern),这是一种创建型设计模式,用于根据输入参数的不同返回不同类的实例,而客户端不需要知道具体类名。文章通过图表类的实例,展示了简单工厂模式的结构、时序图、代码实现、优缺点以及适用环境,并提供了Java代码示例和扩展应用,如通过配置文件读取参数来实现对象的创建。
【一】设计模式~~~创建型模式~~~简单工厂模式(Java)
|
1月前
|
设计模式 uml C语言
设计模式----------工厂模式之简单工厂模式(创建型)
这篇文章详细介绍了简单工厂模式,包括其定义、应用场景、UML类图、通用代码实现、运行结果、实际应用例子,以及如何通过反射机制实现对象创建,从而提高代码的扩展性和维护性。
设计模式----------工厂模式之简单工厂模式(创建型)
|
1月前
|
设计模式 uml
设计模式-------------工厂模式之工厂方法模式(创建型)
工厂方法模式是一种创建型设计模式,它通过定义一个用于创建对象的接口,让子类决定实例化哪一个类,从而实现类的实例化推迟到子类中进行,提高了系统的灵活性和可扩展性。
|
1月前
|
设计模式 测试技术 Go
[设计模式]创建型模式-简单工厂模式
[设计模式]创建型模式-简单工厂模式
|
2月前
|
设计模式 算法 Java
跟着GPT学设计模式之模板模式
模板模式是一种行为型设计模式,它定义了一个操作中的算法骨架,将一些步骤的具体实现延迟到子类中。该模式使得子类可以在不改变算法结构的情况下,重新定义算法中的某些步骤。
44 6
|
1月前
|
设计模式 人工智能 达摩院
设计模式的基础问题之模板模式在软件开发中的优势是什么
设计模式的基础问题之模板模式在软件开发中的优势是什么
|
1月前
|
设计模式 项目管理
设计模式的基础问题之生成器模式在项目管理应用的问题如何解决
设计模式的基础问题之生成器模式在项目管理应用的问题如何解决