美文网首页
java 中的设计模式

java 中的设计模式

作者: 二十_f15f | 来源:发表于2019-10-22 16:11 被阅读0次

    代理模式是什么

    代理模式是一种设计模式,简单说即是在不改变源码的情况下,实现对目标对象的功能扩展。

    比如说有一个买房的对象叫BuyHouser,这个对象有一个方法叫buyhouse();

        public void buyhouse() {
            System.out.println("我要买房");
        }
    

    这是一个公共的买房对象,你不是很喜欢,所以你希望通过自己的方式买房。也就是对目标对象BuyHouser的buyhouser方法进行功能扩展。

        public void buyhouse() {
            System.out.println("买房先准备钱");
            System.out.println("我买房了");
            System.out.println("买完后装修");
        }
    

    就这样可以通过修改相关代码实现上面的需求,但是一些情况下相关的代码是无法修改的,这个时候我们的代理模式就闪亮登场了

    Java中代理模式分为两种

    1. 静态代理
    2. 动态代理

    先说一说静态代理

    1. 先准备一个接口,静态代理要求目标对象和代理对象实现同一个接口
    public interface BuyHouse {
        void buyhouse();
    }
    
    
    1. 然后准备目标对象实现这个接口
    public class YouBuyHouse implements BuyHouse {
        @Override
        public void buyhouse() {
            System.out.println("我要买房");
        }
    }
    
    1. 然后创建我们的代理对象
    public class ProxyBuyHouse implements BuyHouse {
        //创建一个对象用于接收目标对象 
        private BuyHouse buyHouse;
        //这里通过构造器在代理类实例化的时候初始化目标对象
        public ProxyBuyHouse(BuyHouse buyHouse){
            this.buyHouse=buyHouse;
        }
        //这里是代理对象的具体方法
        @Override
        public void buyhouse() {
            System.out.println("卖房先准备钱");
            //这里调用了目标对象的方法(这里是必须的,不然就算不上代理模式了)
            buyHouse.buyhouse();
            System.out.println("买完后装修");
        }
    }
    

    代理已经完成了,然后我们编写一个测试类来看看效果

    public class BuyHouseTest {
        public static void main(String[] args) {
            //创建目标对象
            BuyHouse youBuyHouse = new YouBuyHouse();
            //创建代理对象,并传入目标对象
            BuyHouse proxyBuyHouse = new ProxyBuyHouse(youBuyHouse);
            proxyBuyHouse.buyhouse();
        }
    }
    

    输出结果是这样的

    卖房先准备钱
    我要买房
    买完后装修
    

    总结:其实这里做的事情无非就是,创建一个代理类ProxyBuyHouse,继承了BuyHouse接口并实现了其中的方法。只不过这种实现特意包含了目标对象的方法,正是这种特征使得看起来像是“扩展”了目标对象的方法。如果代理对象中只是简单地对buyhouse()方法做了另一种实现而没有包含目标对象的方法,也就不能算作代理模式了。所以这里的包含是关键。

    缺点:这种实现方式很直观也很简单,但其缺点是代理对象必须提前写出,如果接口层发生了变化,代理对象的代码也要进行维护。如果能在运行时动态地写出代理对象,不但减少了一大批代理类的代码,也少了不断维护的烦恼,不过运行时的效率必定受到影响。这种方式就是接下来的动态代理。

    2. 动态代理(也叫JDK代理)

    这里我们依据按照上面买房的例子进行代理

    public interface BuyHouse {
        void buyhouse();
    }
    
    public class YouBuyHouse implements BuyHouse {
        @Override
        public void buyhouse() {
            System.out.println("我要买房");
        }
    }
    

    动态代理实现起来也很容易

    public class BuyHoseTest2 {
        public static void main(String[] args) {
            //创建目标对象
            YouBuyHouse youBuyHouse = new YouBuyHouse();
            //创建代理对象
            BuyHouse proxy = (BuyHouse) Proxy.newProxyInstance(
                    youBuyHouse.getClass().getClassLoader(),
                    youBuyHouse.getClass().getInterfaces(),
                    new InvocationHandler() {
                        @Override
                        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                            System.out.println("买房先存钱");
                            Object returnValue = method.invoke(youBuyHouse,args);
                            System.out.println("买完房后装修");
                            return returnValue;
                        }
                    }
            );
            proxy.buyhouse();
        }
    }
    

    这里我们先创建了一个目标对象,然后通过 Proxy 类的静态方法newProxyInstance获取代理类对象

    static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces,InvocationHandler h )
    

    接收的三个参数依次为:

    • ClassLoader loader:指定当前目标对象使用类加载器,写法固定
    • Class<?>[] interfaces:目标对象实现的接口的类型,写法固定
    • InvocationHandler h:事件处理接口,需传入一个实现类(即代理类),一般直接使用匿名内部类

    总结:这种方法因为经过jdk的封装所以使用起来比较简单
    缺点:这种和静态代理一样需要实现接口才可以进行相应的操作,如果不想这样可以通过spring 的Cglib代理实现(这个下次写)

    相关文章

      网友评论

          本文标题:java 中的设计模式

          本文链接:https://www.haomeiwen.com/subject/digqvctx.html