zoukankan      html  css  js  c++  java
  • 用最通俗的话说23种设计模式之代理模式

    有一句话这么说“有事秘书干,没事干秘书”。其实生活中秘书就是一种典型的代理。说白了可以这么理解,客户不是因为秘书才谈生意的,而是因为老板。但是老板生意很多,所以用一个秘书来代理事务。所以在百度百科说了这么一句对代理模式的理解:在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用。中介说明了一件事就是代理类本身没有什么实现功能,仅仅是多了一层包装而已。在功能没有直接引用对象那么快。但是这种关系是非常常见的。

    代理模式作为设计模式里面最常用的模式之一所以必须要搞懂。因为代理模式无处不在。在ssh框架中每个框架都使用代理模式。在spring中的aop(面向切面的编程)的实质就是代理模式。

    代理分为两种:一种是静态代理,一种是动态代理。首先明白一点,代理是建立在接口关系上的。

    比如说,我现在很忙但需要买一张汽车票。所以我就可以叫秘书去干。

    我写一个静态代理的例子:

    //这是一个票的接口

    packagecom.fish.Test;

     

    public interface Ticket {

    public void show();

    }

    //这是一个公共汽车票类

    public class BusTicket implements Ticket{

     

        @Override

        public void show() {

            System.out.println("汽车票");

           

        }

     

    }

    //通过代理购票类

    packagecom.fish.Test;

     

    public class MyProxy implements Ticket {

        private Ticket ticket;

     

        public MyProxy(Ticket ticket) {

            super();

            this.ticket = ticket;

        }

     

        @Override

        public void show() {

            ticket.show();

        }

       

        public void buy(){

            System.out.print("代理帮我买了:");

            show();

        }

     

    }

    //测试方法写一个

    public class Test {

    public static void main(String[] args) {

        //先利用多态创建一张票

        Ticket ticket=new BusTicket();

        //将这张票给代理

        MyProxy myProxy=new MyProxy(ticket);

        //代理去买

        myProxy.buy();

    }

    }

    结果显示

    代理帮我买了代理买了:汽车票。

     

    那我用动态代理怎么实现了?其实很简单。首先明白代理类干了什么?说白了就是创建了汽车票类和调用汽车票类的方法。

    那么动态代理就是利用反射来实现的。在java中本身就一个代理类Proxy.这个类里面有几个方法,大家可以看看api我就不说了。我们要实现一个反射方法的句柄对象。然后利用这个类来创建实例。请看代码:其他代码不变就是修改代理类

    packagecom.fish.Test;

     

    importjava.lang.reflect.InvocationHandler;

    importjava.lang.reflect.Method;

    importjava.lang.reflect.Proxy;

     

    public class ProxyDy implements InvocationHandler {

        private Object obj;//从外部接受一个对象

     

       

        public ProxyDy() {

            System.out.println("代理帮我买了:");

        }

    //这是一个绑定的方法。

        public Object bind(Object delegate) {

            this.obj = delegate;//将外部的对象给obj

            return Proxy.newProxyInstance(delegate.getClass().getClassLoader(),

                    delegate.getClass().getInterfaces(),this);//这里java的代理类,看单词意思就是创建一个代理实例。第一个参数,就是获得外部对象的类加载器,第二个就是获得该对象的接口,this是一个InvocationHandler接口的实例,这里不就是this对象自己吗。

        }

     

        @Override

        public Object invoke(Object proxy, Method method, Object[] args)

                throws Throwable {

    //这是一个反射方法的方法你要反射的是obj对象,args里面放了obj的所有方法。

            method.invoke(obj, args);

            return null;

        }

     

    }

    下面写一个测试类

    packagecom.fish.Test;

     

    importjava.lang.reflect.InvocationHandler;

    importjava.lang.reflect.Method;

    import java.lang.reflect.Proxy;

     

    public class Test2 {

        public static void main(String[] args) {

    //创建一个代理

            ProxyDy proxy = new ProxyDy();

    //叫代理买票

            Ticket ticket = (Ticket) proxy.bind(new BusTicket());

    //代理买到票

            ticket.show();

       

     

        }

    }

    结果是:

    代理帮我买了一张汽车票

    其实动态代理不难,上面两个类的代码我可以用一段代码实现

        final Ticket ticket = new BusTicket();

     

            Ticket ticket2 = (Ticket) Proxy.newProxyInstance(ticket.getClass()

                    .getClassLoader(),ticket.getClass().getInterfaces(),

                    new InvocationHandler() {

     

                        @Override

                        public Object invoke(Object proxy, Method method,

                                Object[] args) throws Throwable {

                            method.invoke(ticket,null);

                            return null;

                        }

                    });

     

            Ticket2.show();

       

    这段代码就能实现上面两个类的代码。我想说的只有一点,动态代理就是通过java本身的Prxoy类实现的。其实我们也可以用反射实现自己的动态代理。

  • 相关阅读:
    nodejs入门API之http模块
    nodejs入门API之fs模块
    编程官方文档中的方法参数格式的含义
    vs Code编辑器智能提示功能
    nodejs入门之模块
    git的安装与使用
    TypeScript入门九:TypeScript的模块
    TypeScript入门八:TypeScript的命名空间
    TypeScript入门七:TypeScript的枚举
    TypeScript入门六:TypeScript的泛型
  • 原文地址:https://www.cnblogs.com/james1207/p/3271070.html
Copyright © 2011-2022 走看看