Java设计模式(二) JDK动态代理

1.什么是代理模式?

    假如你的公司是一家软件公司,你是一位软件工程师。客户带着需求去找公司显然不会直接去找你,而是去找商务,此时客户会认为商务就代表公司

客户 ——>商务(代理对象)——>软件工程师(真实对象)

综上,商务和软件工程师就是代理和被代理的关系,客户是经过商务去访问软件工程师的,商务就是代理对象,软件工程师真实对象。我们需要在调用者调用对象之前产生一个代理对象,而这个代理对象需要和真实的对象建立代理关系。代理必须分为两步:

  • 代理对象和真实对象建立代理关系
  • 实现代理对象的代理逻辑方法

2.JDK动态代理

       JDK动态代理是由java.lang.reflect.*包提供的。它必须借助一个接口才能产生代理对象,因此首先编写一个接口和它的实现类。

//必须经过接口才能产生代理对象
public interface LaoZong {
    public void eat();
    public void sleep();
}

public class LaoZongImpl implements LaoZong {
    public void eat() {
        System.out.println("吃饭");
    }

    public void sleep() {
        System.out.println("睡觉!");
    }
}

   这里的LaoZongImpl 就是真正要访问的对象。

   有了接口和实现类就可以开始动态代理了。使用JDK动态代理必须要去实现java.lang.reflect.InvocationHandler,它里面只有一个invoke方法,并提供接口数组用于下挂代理对象。

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
 * **这就是一个代理对象
 *  建立起真实对象和代理对象之间的关系
 */
public class Mishu implements InvocationHandler {

    //真实的对象
    private Object target;

    /**
     * 建立代理对象和真实对象之间的关系,返回代理对象
     * @param target 真实的对象
     * @return 代理对象
     */
    public Object bind(Object target){
        this.target = target;
        return Proxy.newProxyInstance( target.getClass().getClassLoader(),
                target.getClass().getInterfaces(),
                this);
    }
    /**
     *  代理逻辑方法
     * @param proxy 代理对象,就是bind生成的对象
     * @param method    当前调度方法
     * @param args  当前方法参数
     * @return  代理结果返回
     * @throws Throwable
     */
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("进入代理逻辑方法");
        System.out.println("在调度真实对象之前的服务");

        Object object = method.invoke( target,args );//相当于调用eat和sleep方法
        System.out.println(method);
        System.out.println("在调度真实对象之后的服务");
        return object;
    }
}

第一步,建立代理和真实对象的关系。这里使用bind方法完成的。这里的target就是真实对象。通过以下代码建立代理对象。

Proxy.newProxyInstance( target.getClass().getClassLoader(),
                target.getClass().getInterfaces(),
                this);

这里的newProxyInstance方法包含三个参数

  • 第一个是类加载器,采用target本身的类加载器
  • 第二个是把生成的动态代理对象挂在哪些接口下。这个写法就是挂在target实现的接口下。
  • 带三个是实现方法逻辑的代理类,this表示当前对象,它必须实现InvocationHandler接口的Invoke方法,它就是代理逻辑方法的实现方法。

第二步 实现代理逻辑方法。invoke方法可以实现代理逻辑。

 public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        
        Object object = method.invoke( target,args );//相当于调用eat和sleep方法
        
        return object;
    }
  • proxy,代理对象,bind方法生成的对象
  • method,当前调度的方法
  • args,调度方法的参数
 Object object = method.invoke( target,args )

上面这串代码相当于调度真实对象的方法,只不过是使用的反射而已。
测试代码

public class Test {
    public static void main(String[] args) {
        Mishu mishu = new Mishu();
        //绑定关系,因为挂在接口LaoZong下,所以声明代理对象LaoZong proxy
        LaoZong proxy = (LaoZong) mishu.bind( new LaoZongImpl() );
        
        //此时LaoZong对象已经是一个代理对象,它会进入代理的逻辑方法invoke()里
        proxy.sleep();
    }
}

输出结果:

进入代理逻辑方法
在调度真实对象之前的服务
睡觉!
public abstract void com.wanfei.service.LaoZong.sleep()
在调度真实对象之后的服务
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值