代理模式 - Java实现

代理模式

代理模式概念

为其他对象提供一个代理以控制对某个对象的访问。代理类主要负责为委托类(真实对象)预处理消息、过滤消息、传递消息给委托类,代理类不现实具体服务,而是利用委托类来完成服务,并将执行结果封装处理。

其实就是代理类为被代理类预处理消息、过滤消息并在此之后将消息转发给被代理类,之后还能进行消息的后置处理。代理类和被代理类通常会存在关联关系(即上面提到的持有的被代理对象的引用),代理类本身不实现服务,而是通过调用被代理类中的方法来提供服务。

静态代理

创建一个接口,然后创建被代理的类实现该接口并且实现该接口中的抽象方法。之后再创建一个代理类,同时使其也实现这个接口。在代理类中持有一个被代理对象的引用,而后在代理类方法中调用该对象的方法。

接口:

public interface HelloInterface {
    void sayHello();
}

被代理类:

public class Hello implements HelloInterface{
    @Override
    public void sayHello() {
        System.out.println("Hello zhanghao!");
    }
}

代理类:

public class HelloProxy implements HelloInterface{
    private HelloInterface helloInterface = new Hello();
    @Override
    public void sayHello() {
        System.out.println("Before invoke sayHello" );
        helloInterface.sayHello();
        System.out.println("After invoke sayHello");
    }
}

代理类调用:被代理类被传递给了代理类HelloProxy,代理类在执行具体方法时通过所持用的被代理类完成调用。

    public static void main(String[] args) {
        HelloProxy helloProxy = new HelloProxy();
        helloProxy.sayHello();
    }
    
输出:
Before invoke sayHello
Hello zhanghao!
After invoke sayHello

使用静态代理很容易就完成了对一个类的代理操作。但是静态代理的缺点也暴露了出来:由于代理只能为一个类服务,如果需要代理的类很多,那么就需要编写大量的代理类,比较繁琐。

动态代理

利用反射机制在运行时创建代理类。

动态代理:

  • 特点:字节码随用随创建,随用随加载

  • 作用:不修改源码的基础上对方法增强

  • 分类:

    • 基于接口的动态代理
    • 基于子类的动态代理

如何创建代理对象:

前提配置:

接口类:

public interface IProducer {

     void saleProduct(float money);

     void afterService(float money);
}

实现类:( 被代理类 )

public class Producer implements IProducer {
    public void saleProduct(float money){
        System.out.println("拿钱销货 " + money);
    }

    public void afterService(float money){
        System.out.println("售后服务 " + money);
    }
}

基于接口的动态代理

使用Proxy类中的newProxyInstance方法 | 提供者:JDK

创建代理对象的要求:被代理类最少实现一个接口,如果没有则不能使用

方法的参数:

  • ClassLoader:类加载器。

    ​ 用于加载代理对象的字节码,和被代理对象使用相同的类加载器。

    ​ 固定写法:被代理类.getClass().getClassLoader()

  • Class[]:字节码数组

    ​ 用于让代理对象和被代理对象有相同方法

    ​ 固定写法:被代理类.getClass().getInterfaces()

  • InvocationHandler:用于提供增强的代码

    ​ 描述如何代理。一般都是写一个该接口的实现类,通常情况下都是匿名内部类,但不是必须。

    ​ 此接口的实现类是谁用谁写

代码实现:

public class Client {

    public static void main(String[] args) {
        final Producer producer = new Producer();
		
        //基于接口的动态代理
        IProducer proxyProducer = (IProducer) Proxy.newProxyInstance(
            producer.getClass().getClassLoader(),
                producer.getClass().getInterfaces(), new InvocationHandler() {
                    /**
                     * 作用:执行被代理对象的任何接口方法都会经过该方法
                     * @param proxy  代理对象的引用
                     * @param method 当前执行的方法
                     * @param args   当前执行方法所需的参数
                     * @return       和被代理对象有相同的返回值
                     * @throws Throwable
                     */
                    public Object invoke(Object proxy, Method method, 
                                         Object[] args) throws Throwable {
//                      return method.invoke(producer,args);//什么都没干

                        //提供增强的代码
                        Object returnValue = null;
                        //1、获取方法执行的参数
                        float money = (Float) args[0];
                        //2、判断当前方法是不是销售
                        if("saleProduct".equals(method.getName())){
                            returnValue = method.invoke(producer, money*0.8f);
                        }
                        return returnValue;
                    }
                });

        proxyProducer.saleProduct(1000f);
    }
}

基于子类的动态代理

使用的类:Enhancer | 提供者:第三方cglib库 | 创建代理对象:Enhancer 类中的 create方法

要求:被代理类不能是最终类

方法的参数:

  • Class : 字节码

    ​ 用于指定被代理对象的字节码

  • Callback:

    ​ 用于提供增强的代码

    ​ 一般使用的都是Callback 接口的子接口实现类:MethodInterceptor (翻译:方法拦截)

MethodInterceptor

public class Client {

    public static void main(String[] args) {
        final Producer producer = new Producer();
        
        //基于子类的动态代理
        Producer cglibProducer = (Producer)Enhancer.create(
            producer.getClass(), new MethodInterceptor() {
            /**
             * 执行被代理对象的任何方法都会经过该方法
             * @param proxy
             * @param method
             * @param args
             *       以上三个参数和基于接口中的动态代理中 invoke 方法的参数是一样的
             * @param methodProxy 当前执行方法的代理对象
             * @return
             * @throws Throwable
             */
            public Object intercept(Object proxy, Method method, 
                                    Object[] args, MethodProxy methodProxy) 
                throws Throwable {
                //提供增强的代码
                Object returnValue = null;
                //1、获取方法执行的参数
                float money = (Float) args[0];
                //2、判断当前方法是不是销售
                if ("saleProduct".equals(method.getName())) {
                    returnValue = method.invoke(producer, money * 0.8f);
                }
                return returnValue;
            }
        });

        cglibProducer.saleProduct(10000f);
    }
}

基于子类的动态代理不需要被代理类继承一个接口

posted @ 2020-06-30 23:49  xtLLL  阅读(133)  评论(0)    收藏  举报