Java设计模式—代理模式

简介: Java设计模式—代理模式

代理就是真实对象的代表。


代理模式的应用场景:


1.远程代理,也就是为一一个对象在不同的地址空间提供局部代表。这样可以隐藏一个对象存在于不同地址空间的事实。


例如当某一个产品在当地做大后,那么就可以去其他地方发展,每个地方可以雇佣一个代理来帮忙打理。


2.虚拟代理,是根据需要创建开销很大的对象。通过它来存放实例化需要很长时间的真实对象。这样就可以达到性能的最优化,比如说你打开一个很大的HTML网页时,里面可能有很多的文字和图片,但你还是可以很快打开它,此时你所看到的是所有的文字,但图片却是一张一张地下载后才能看到。那些未打开的图片框,就是通过虚拟代理来替代了真实的图片,此时代理存储了真实图片的路径和尺寸。

3.安全代理,用来控制真实对象访问时的权限。一般用于对象应该有不同的访问权限的时候。


4.智能引用代理(项目开发中应用最广泛),是指当调用真实的对象时,代理处理另外一些事。如计算真实对象的引用次数,  这样当该对象没有引用时,可以自动释放它;或当第一次引用- 一个持久对象时,将它装入内存;或在访问一个实际对象前,检查是否已经锁定它,以确保其他对象不能改变它。它们都是通过代理在访问一个对象时附加一些内务处理。(日志管理,权限管理,事物处理...)


代理模式结构图


20180918112227404 (1).png

静态代理:代理和被代理对象在代理之前是确定的。他们都实现相同的接口或者继承相同的抽象类。


常用的静态代理方式通过聚合和继承来完成,聚合是通过实现相同的接口来实现的,继承是通过继承相同的抽象类来实现的。


聚合是比继承更适合代理模式:


如果使用继承来实现代理功能的叠加,每当我们要添加或者更改功能时,我们就需要写不同的类来实现功能的变更,就算功能实现的顺序改变也需要重新写一个代理类。这样代理类会越来越多,使得代码臃肿。


使用聚合的方式来就可以灵活调用各个代理模块,代理之间可以互相传递。


静态代理


代理类对真实对象需要做的事情前后添加东西,客户端通过接口去实现代理类,代理类又依赖目标类

public class Main {
    public static void main(String[] args) {
        Subject subject=new Proxy(new RealSubject());
        subject.execute();
    }
}
//对需要做的一件事进行抽象
interface Subject {
    void execute();
}
//真实对象
class RealSubject implements Subject {
    @Override
    public void execute() {
        // TODO Auto-generated method stub
        System.out.println("吃饭...");
        try {
            Thread.sleep(500);
        } catch (InterruptedException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
}
//代理对象
class Proxy implements Subject {
    //真实对象
    RealSubject realSubject;
    public Proxy(RealSubject realSubject) {
        super();
        this.realSubject = realSubject;
    }
    @Override
    public void execute() {
        // TODO Auto-generated method stub
        long starttime = System.currentTimeMillis();
        System.out.println("记录开始时间....");
        realSubject.execute();
        long endtime = System.currentTimeMillis();
        System.out.println("记录结束时间....  用时:"
                + (endtime - starttime) + "毫秒!");
    }
}

当我们有多个接口,多个实现类,这些实现类都需要使用到记录时间的功能,如果我们使用静态代理的话,需要为每个实现类去写一个代理接口,这样先让会让代码变得臃肿起来。这时我们就可以使用动态代理的方式来处理这个问题。


动态代理分为JDK动态代理和Cglib动态代理(本文主要说的是Jdk动态代理)


Jdk动态代理只能代理实现接口的类,没有实现接口的类不能实现JDK的动态代理。


Cglib动态代理是针对类来实现代理的,对指定目标类产生一个子类,通过方法拦截技术拦截所有父类方法的调用。


Cglib看https://blog.csdn.net/BushQiang/article/details/98260252

Java动态代理类位于java.lang.reflect包下,一-般主要涉及到以下两个类:


(1)Interface InvocationHandler :该接口中仅定义了一个方法public object invoke(Object obj;Method method, Object[] args)在实际使用时,第一个参数obj一般是指代理类, method是被代理的方法, args为该方法的参数数组。 这个抽象方法在代理类中动态实现。


(2)Proxy :该类即为动态代理类

static Object newProxyInstance(ClassLoader loader, Class[] interfaces,InvocationHandler h) :返回代理类的一个实例,返回后的代理类可以当作被代理类使用(可使用被代理类的在接口中声明过的方法)


上面的时间代理可以修改为适合任何类的

package com.smxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
 * 
 * @Description :动态代理模式
 * @author Bush罗
 * @date 2018年10月7日
 *
 */
public class ProxyTest {
  public static void main(String[] args) throws InstantiationException, IllegalAccessException {
    Work work = (Work) new ProxyTime().getInstance(new RealProxy());
    work.execute();
  }
}
interface Work {
  void execute();
}
class RealProxy implements Work {
  @Override
  public void execute() {
    // TODO Auto-generated method stub
    System.out.println("吃饭...");
  }
}
class ProxyTime implements InvocationHandler {
  private Work target;
  public Object getInstance(Work target) {
    this.target = target;
    Class<?> clazz = target.getClass();
    Object obj = Proxy.newProxyInstance(clazz.getClassLoader(),
        clazz.getInterfaces(), this);
    return obj;
  }
  @Override
  public Object invoke(Object obj, Method method, Object[] args)
      throws Throwable {
    long starttime = System.currentTimeMillis();
    System.out.println("记录开始时间....");
    // 第一个参数是target,也就是被代理类的对象;第二个参数是方法中的参数
    method.invoke(target, args);
    long endtime = System.currentTimeMillis();
    System.out.println("记录结束时间....  用时:" 
        + (endtime - starttime) + "毫秒!");
    return null;
  }
}
相关文章
|
2月前
|
设计模式 缓存 Java
「全网最细 + 实战源码案例」设计模式——代理模式
代理模式(Proxy Pattern)是一种结构型设计模式,通过代理对象控制对目标对象的访问并添加额外功能。它分为静态代理和动态代理,后者包括JDK动态代理和CGLIB动态代理。JDK动态代理基于接口反射生成代理类,而CGLIB通过继承目标类生成子类。代理模式适用于延迟初始化、访问控制、远程服务、日志记录和缓存等场景,优点是职责分离、符合开闭原则和提高安全性,缺点是增加系统复杂性。
80 25
|
3月前
|
设计模式 前端开发 数据安全/隐私保护
前端必须掌握的设计模式——代理模式
代理模式(Proxy Pattern)是一种结构型设计模式,通过引入“替身”对象来间接访问真实对象,从而解耦并提升性能和安全性。例如,知名艺人复出后,经纪人作为代理筛选商单,确保只处理符合团队利益的请求。代码实现中,定义接口`IService`,艺人和经纪人都实现该接口,经纪人在访问时进行过滤和转发。代理模式常用于权限控制、性能优化等场景,如前端中的Tree-shaking和ES6的Proxy构造方法。
前端必须掌握的设计模式——代理模式
|
4月前
|
设计模式 消息中间件 搜索推荐
Java 设计模式——观察者模式:从优衣库不使用新疆棉事件看系统的动态响应
【11月更文挑战第17天】观察者模式是一种行为设计模式,定义了一对多的依赖关系,使多个观察者对象能直接监听并响应某一主题对象的状态变化。本文介绍了观察者模式的基本概念、商业系统中的应用实例,如优衣库事件中各相关方的动态响应,以及模式的优势和实际系统设计中的应用建议,包括事件驱动架构和消息队列的使用。
|
4月前
|
设计模式 Java 数据库连接
Java编程中的设计模式:单例模式的深度剖析
【10月更文挑战第41天】本文深入探讨了Java中广泛使用的单例设计模式,旨在通过简明扼要的语言和实际示例,帮助读者理解其核心原理和应用。文章将介绍单例模式的重要性、实现方式以及在实际应用中如何优雅地处理多线程问题。
66 4
|
5月前
|
Java Spring 数据库连接
[Java]代理模式
本文介绍了代理模式及其分类,包括静态代理和动态代理。静态代理分为面向接口和面向继承两种形式,分别通过手动创建代理类实现;动态代理则利用反射技术,在运行时动态创建代理对象,分为JDK动态代理和Cglib动态代理。文中通过具体代码示例详细讲解了各种代理模式的实现方式和应用场景。
92 0
[Java]代理模式
|
5月前
|
设计模式 Java 程序员
[Java]23种设计模式
本文介绍了设计模式的概念及其七大原则,强调了设计模式在提高代码重用性、可读性、可扩展性和可靠性方面的作用。文章还简要概述了23种设计模式,并提供了进一步学习的资源链接。
104 0
[Java]23种设计模式
|
4月前
|
设计模式 JavaScript Java
Java设计模式:建造者模式详解
建造者模式是一种创建型设计模式,通过将复杂对象的构建过程与表示分离,使得相同的构建过程可以创建不同的表示。本文详细介绍了建造者模式的原理、背景、应用场景及实际Demo,帮助读者更好地理解和应用这一模式。
142 0
|
5月前
|
设计模式 监控 算法
Java设计模式梳理:行为型模式(策略,观察者等)
本文详细介绍了Java设计模式中的行为型模式,包括策略模式、观察者模式、责任链模式、模板方法模式和状态模式。通过具体示例代码,深入浅出地讲解了每种模式的应用场景与实现方式。例如,策略模式通过定义一系列算法让客户端在运行时选择所需算法;观察者模式则让多个观察者对象同时监听某一个主题对象,实现松耦合的消息传递机制。此外,还探讨了这些模式与实际开发中的联系,帮助读者更好地理解和应用设计模式,提升代码质量。
Java设计模式梳理:行为型模式(策略,观察者等)
|
6月前
|
存储 设计模式 安全
Java设计模式-备忘录模式(23)
Java设计模式-备忘录模式(23)
|
6月前
|
设计模式 存储 算法
Java设计模式-命令模式(16)
Java设计模式-命令模式(16)