代理模式

Posted zad27

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了代理模式相关的知识,希望对你有一定的参考价值。

一 代理模式

1.1 代理模式介绍

定义:代理模式(Proxy Pattern)也称为委托模式,是为其他对象提供一种代理以控制对这个对象的访问的模式.

主要解决:在直接访问对象时带来的问题,比如说:要访问的对象在远程的机器上。在面向对象系统中,有些对象由于某些原因(比如对象创建开销很大,或者某些操作需要安全控制,或者需要进程外的访问),直接访问会给使用者或者系统结构带来很多麻烦,我们可以在访问此对象时加上一个对此对象的访问层。

何时使用:想在访问一个类时做一些控制。

如何解决:增加中间层。

关键代码:实现与被代理类组合。

应用实例: 1、明星经纪人  2、我爱我家出租房屋中介  3、代买我车票 4、一张支票或银行存单是账户中资金的代理。支票在市场交易中用来代替现金,并提供对签发人账号上资金的控制。 5、spring aop。

优点: 1、职责清晰。 2、高扩展性。 3、智能化。

缺点: 1、由于在客户端和真实主题之间增加了代理对象,因此有些类型的代理模式可能会造成请求的处理速度变慢。 2、实现代理模式需要额外的工作,有些代理模式的实现非常复杂。

使用场景:按职责来划分,通常有以下使用场景: 1、远程代理。 2、虚拟代理。 3、Copy-on-Write 代理。 4、保护(Protect or Access)代理。 5、Cache代理。 6、防火墙(Firewall)代理。 7、同步化(Synchronization)代理。 8、智能引用(Smart Reference)代理。

注意事项: 1、和适配器模式的区别:适配器模式主要改变所考虑对象的接口,而代理模式不能改变所代理类的接口。 2、和装饰器模式的区别:装饰器模式为了增强功能,而代理模式是为了加以控制。

1.2 代理模式代码实现

代理模式种有2种角色,代理者(ProxyImage)与被代理者(RealImage),代理者将获得被代理者所有信息,才能对被代理者进行‘额外‘加工.

技术分享图片

1. Image接口

public interface Image {
   void display();
}

2. RealImage类

public class RealImage implements Image {
 
   private String fileName;
 
   public RealImage(String fileName){
      this.fileName = fileName;
      loadFromDisk(fileName);
   }
 
   @Override
   public void display() {
      System.out.println("Displaying " + fileName);
   }
 
   private void loadFromDisk(String fileName){
      System.out.println("Loading " + fileName);
   }
}

3. ProxyImage类

public class ProxyImage implements Image{
 
   private RealImage realImage;
   private String fileName;
 
   public ProxyImage(String fileName){
      this.fileName = fileName;
   }
 
   @Override
   public void display() {
      if(realImage == null){
         realImage = new RealImage(fileName);
      }
      realImage.display();
   }
}

3. 测试类

public class ProxyPatternDemo {
   
   public static void main(String[] args) {
      Image image = new ProxyImage("test_10mb.jpg");
 
      // 图像将从磁盘加载
      image.display(); 
      System.out.println("");
      // 图像不需要从磁盘加载
      image.display();  
   }
}

4. 测试结果

Loading test_10mb.jpg
Displaying test_10mb.jpg

Displaying test_10mb.jpg

 

 

二 JDK动态代理CGLIB代理对比

2.1 JDK动态代理与CGLIB原理区别

java动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理。

而cglib动态代理是利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。

JDK动态代理和CGLIB字节码生成的区别?

 (1)JDK动态代理只能对实现了接口的类生成代理,而不能针对类
 (2)CGLIB是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法
   因为是继承,所以CGLIB不能对声明为final的类生成代理类

2.2 2种代理代码实现

被代理对象接口

public interface PersonManager {
    
    void addPerson(String id, String password);

    void delPerson(String id);
}

被代理对象

public class PersonManagerImpl implements PersonManager {
    @Override
    public void addPerson(String id, String password) {
        System.out.println("添加用户成功");
    }

    @Override
    public void delPerson(String id) {
        System.out.println("删除用户成功");
    }
}

 

JDK动态代理类

public class JDKProxyDemo implements InvocationHandler {

    private Object targetObj;

    public Object newProxy(Object obj) {
        targetObj = obj;
        return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), this);
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        checkParam();
        return method.invoke(targetObj, args);
    }

    private void checkParam() {
        System.out.println("param is OK");
    }
}

 

CGLIBProxy动态代理类

 

public class CGLibProxyDemo implements MethodInterceptor {
    private Object targetObject;

    public Object newProxy(Object obj) {
        this.targetObject = obj;
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(obj.getClass());
        enhancer.setCallback(this);
        Object proxyObj = enhancer.create();        
        return proxyObj;
    }

    @Override
    public Object intercept(Object proxy, Method method, Object[] args,
                            MethodProxy methodProxy) throws Throwable {

        checkParam();
        return method.invoke(targetObject, args);
    }

    private void checkParam() {
        System.out.println("param is OK");
    }
}

 

测试类

class ReflectTest {

    private PersonManager personManager;

    @BeforeEach
    public void init() {
        personManager = new PersonManagerImpl();
    }

    @Test
    void jdkProxyTest() {
        PersonManager proxy = (PersonManager) new JDKProxyDemo().newProxy(personManager);

        System.out.println("=========添加用户测试开始=========");
        proxy.addPerson("1", "111111");
        System.out.println("=========添加用户测试结束=========");
        System.out.println();

        System.out.println("=========删除用户测试开始=========");
        proxy.delPerson("1");
        System.out.println("=========删除用户测试结束=========");

    }

    @Test
    void cglibProxyTest() {
        PersonManager proxy = (PersonManager) new CGLibProxyDemo().newProxy(personManager);
        System.out.println("=========添加用户测试开始=========");
        proxy.addPerson("1", "111111");
        System.out.println("=========添加用户测试结束=========");
        System.out.println();

        System.out.println("=========删除用户测试开始=========");
        proxy.delPerson("1");
        System.out.println("=========删除用户测试结束=========");
    }
}

 

参考资料:

1. http://www.runoob.com/design-pattern/proxy-pattern.html

2. https://www.cnblogs.com/waves-a/p/8036857.html



以上是关于代理模式的主要内容,如果未能解决你的问题,请参考以下文章

scrapy按顺序启动多个爬虫代码片段(python3)

用于从 cloudkit 检索单列的代码模式/片段

java代码实现设计模式之代理模式

代理模式(静态代理动态代理)代码实战(详细)

Java设计模式-代理模式之动态代理(附源代码分析)

代理模式(静态代理)