Java代理模式(静态代理&动态代理)

一、概述

  定义:为其他对象提供一种代理,以控制这个对象的访问。

  代理模式的种类:静态代理和动态代理

二、静态代理

  ps:我们创建一个Book用于基础操作,再创建一个JavaBook用来代理Book类的功能。

  1.Book.java

package com.yw.reflectjavalib.proxy.staticproxy;

/**
 * 定义一个book实体
 * create by yangwei
 * on 2020-02-16 17:48
 */
public class Book {
    public void doWork() {
        System.out.println("读书");
    }
}

  2.JavaBook.java

package com.yw.reflectjavalib.proxy.staticproxy;

/**
 * java实体
 * create by yangwei
 * on 2020-02-16 17:49
 */
public class JavaBook {
    private Book book;

    public JavaBook(Book book) {
        this.book = book;
    }

    public void doWork() {
        System.out.println("Java");
        book.doWork();
    }
}

  3.main方法测试

package com.yw.reflectjavalib.proxy.staticproxy;

/**
 * 静态代理测试
 * create by yangwei
 * on 2020-02-16 17:47
 */
public class StaticProxyDemo {
    public static void main(String[] args) {
        Book book = new Book();
        //这里由于Javabook中有Book类的引用,所以JavaBook除了可以做自己的事情外,可以顺便把Book的工作给做了。
        //也就是JavaBook代理了Book了的所有功能。这就是静态代理,非常的简单。
        //通常这种模式也可以延伸成为装饰模式,即JavaBook把Book类给装饰的更强大了。
        JavaBook javaBook = new JavaBook(book);
        javaBook.doWork();
    }
}

  

三、动态代理

  在上面的例子中我们虽然使用JavaBook代理了Book类,但是仔细一想这样做是有问题的。例如:每一个类都需要有一个对应的Proxy类,随着类增多则代理类也会增加很多。下面介绍一种只需要一个代理类就能搞定的方法——动态代理。

   在java.lang.reflect包中有一个Proxy类。Proxy类的Proxy.newProxyInstance(ClassLoader,Class[] interfaces,InvocationHandler)方法可以将目标对象进行注入,并实现对目标对象的修改。

  参数介绍:

  1.ClassLoader classLoader 目标对象的classloader

  2.Class<?>[] interfaces 目标对象实现的接口类型

  3.InvocationHandler handler 一个实现了InvocationHandler的对象,通过构造方法把目标对象注入,并在这个自定义类中做一些自定义的操作。

下面通过一个简单的例子玩一下动态代理,加深对动态代理的理解。ps:输出水果的颜色。

  1.IFruits.java

  

package com.yw.reflectjavalib.proxy.staticproxy.dnyproxy;

/**
 * 定义一个水果的接口
 * create by yangwei
 * on 2020-02-16 18:10
 */
public interface IFruits {
    /**
     * 水果的颜色
     */
    String getColor();
}

  

  2.Apple.java

package com.yw.reflectjavalib.proxy.staticproxy.dnyproxy;

/**
 * create by yangwei
 * on 2020-02-16 18:12
 */
public class Apple implements IFruits {
    @Override
    public String getColor() {
        return "红色";
    }
}

  

  3.MyInvocationHandler.java

package com.yw.reflectjavalib.proxy.staticproxy.dnyproxy;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

/**
 * create by yangwei
 * on 2020-02-16 18:13
 */
public class MyInvocationHandler implements InvocationHandler {
    private Object target;

    public MyInvocationHandler(Object obj) {
        this.target = obj;
    }

    @Override
    public Object invoke(Object o, Method method, Object[] objects) throws Throwable {
        System.out.print("水果的颜色:");
        return method.invoke(target, objects);
    }
}

  

  4.DnyProxyDemo.java

package com.yw.reflectjavalib.proxy.staticproxy.dnyproxy;

import java.lang.reflect.Proxy;

/**
 * 动态代理测试
 * create by yangwei
 * on 2020-02-16 18:01
 */
public class DnyProxyDemo {
    /**
     * 动态代理
     *
     * @param args
     */
    public static void main(String[] args) {
        //创建一个Apple实例
        IFruits iFruits = new Apple();
        //动态代理Apple,代理后把代理对象返回,由于是面向接口编程,所以可以直接把返回结果转换为接口类型。
        //而MyInvocationHandler中已经对代理对象apple做了相关的操作。例如:加上了"水果的颜色"
        //所以打印出来的值为:水果的颜色为:红色
        //如果去掉代理类,那打印结果只能是:红色。
        //这就是动态带来的神奇之处,只要把目标代理对象传进去,就可以对目标代理对象做一些个性化的包装
        //动态代理相较于静态代理的优势是,我们不必为每一个对象都创建一个代理对象了。直接使用Proxy.newProxyInstance弄出来一个就OK了。
        IFruits object = (IFruits) Proxy.newProxyInstance(iFruits.getClass().getClassLoader(), iFruits.getClass().getInterfaces(), new MyInvocationHandler(iFruits));
        System.out.println(object.getColor());
    }
}

  以上就是动态代理的完整例子,在以上例子中,如果我们不使用动态代理,直接处处apple的颜色,只能得到“红色”。使用动态代理后会在水果颜色前面加上描述:“水果的颜色为:红色”,大家可以运行下试试,感受下。

猜你喜欢

转载自www.cnblogs.com/tony-yang-flutter/p/12317983.html