java 中的设计模式

代理模式是什么

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

比如说有一个买房的对象叫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 中的设计模式)