• 代理模式


    一 代理模式

    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

  • 相关阅读:
    ORA-65114
    Mariadb 10.14 mysqldump error: 1049
    nginx:403 forbidden
    ORA-01017
    oracle 12C 之 Clone 数据库
    Selinux的基本使用
    This system is not registered to Red Hat Subscription Management
    Emacs: too long for unix domain socket
    hive 之 元数据表结构(Mysql)
    hive之SerDe
  • 原文地址:https://www.cnblogs.com/zad27/p/9899097.html
Copyright © 2020-2023  润新知