第四节 常用设计模式之代理模式

亮子 2021-09-10 18:27:14 10782 0 0 0

6、代理模式

代理(Proxy)是一种设计模式,提供了对目标对象另外的访问方式,即通过代理对象访问目标对象。这样做的好处是:可以在目标对象实现的基础上,增强额外的功能操作,即扩展目标对象的功能。
这里使用到编程中的一个思想:不要随意去修改别人已经写好的代码或者方法,如果需改修改,可以通过代理的方式来扩展该方法。

举个例子来说明代理的作用:假设我们想邀请一位明星,那么并不是直接连接明星,而是联系明星的经纪人,来达到同样的目的。明星就是一个目标对象,他只要负责活动中的节目,而其他琐碎的事情就交给他的代理人(经纪人)来解决,这就是代理思想在现实中的一个例子。

1)、静态代理

静态代理在使用时,需要定义接口或者父类,被代理对象与代理对象一起实现相同的接口或者是继承相同父类。

(1)定义接口

public interface IMarry {
    void doMarry();
}

(2)目标对象

public class MarryPerson implements IMarry {

    public void doMarry() {
        System.out.println("!!!举行婚礼了!!!");
    }
}

(3)代理对象

package com.shenmazong.proxy;

/**
 * @program: server-java-demo
 * @description: MarryPersonProxy
 * @author: 亮子说编程
 * @create: 2020-10-15 12:06
 **/
public class MarryPersonProxy implements IMarry {

    private IMarry target;

    public MarryPersonProxy(MarryPerson person) {
        this.target = person;
    }

    public void doMarry() {
        doUnknow();
        doLove();
        target.doMarry();
        doLeave();
        doUnknow();
    }

    private void doUnknow() {
        System.out.println("我们是陌生人");
    }

    private void doLove() {
        System.out.println("我们相爱了");
    }

    private void doLeave() {
        System.out.println("我们离婚了");
    }
}

(4)执行静态代理

public class MarryApplication {

    public static void main(String[] args) {
        MarryPerson marryPerson = new MarryPerson();
        MarryPersonProxy marryPersonProxy = new MarryPersonProxy(marryPerson);
        marryPersonProxy.doMarry();
    }
}

静态代理总结:
优点:

  • 可以做到在不修改目标对象的功能前提下,对目标功能扩展。

缺点:

  • 因为代理对象需要与目标对象实现一样的接口,所以会有很多代理类,类太多
  • 同时,一旦接口增加方法,目标对象与代理对象都要维护

2)、动态代理

动态代理有以下特点:

  • 代理对象,不需要实现接口
  • 代理对象的生成,是利用JDK的API,动态的在内存中构建代理对象(需要我们指定创建代理对象/目标对象实现的接口的类型)
  • 动态代理也叫做:JDK代理,接口代理

JDK中生成代理对象的API
代理类所在包:java.lang.reflect.Proxy
JDK实现代理只需要使用newProxyInstance方法,但是该方法需要接收三个参数,完整的写法是:

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

注意该方法是在Proxy类中是静态方法,且接收的三个参数依次为:

  • ClassLoader loader,:指定当前目标对象使用类加载器,获取加载器的方法是固定的
  • Class<?>[] interfaces,:目标对象实现的接口的类型,使用泛型方式确认类型
  • InvocationHandler h:事件处理,执行目标对象的方法时,会触发事件处理器的方法,会把当前执行目标对象的方法作为参数传入

总结:

  • 利用反射机制在运行时创建代理类。
  • 接口、被代理类不变,我们构建一个handler类来实现InvocationHandler接口。

(1)定义接口

public interface IMarry {
    void doMarry();
}

(2)目标对象

public class MarryPerson implements IMarry {

    public void doMarry() {
        System.out.println("!!!举行婚礼了!!!");
    }
}

(3)构建handler类

public class MarryInvocationHandler implements InvocationHandler {

    Object target;

    public MarryInvocationHandler(Object target) {
        this.target = target;
    }

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

        System.out.println("Before invoke "  + method.getName());
        method.invoke(target, args);
        System.out.println("After invoke " + method.getName());

        return null;
    }
}

(4)执行动态代理

public class MarryInvocationApplication {

    public static void main(String[] args) {
        // 是否保存生成的代理类class文件,默认false不保存
        System.getProperties().setProperty("sun.misc.ProxyGenerator.saveGeneratedFiles", "true");

        IMarry marry = new MarryPerson();

        InvocationHandler handler = new MarryInvocationHandler(marry);

        IMarry proxyHello = (IMarry) Proxy.newProxyInstance(marry.getClass().getClassLoader(), marry.getClass().getInterfaces(), handler);

        proxyHello.doMarry();
    }
}