全国咨询/投诉热线:400-618-4000

Java培训实战教程之Java的动态代理

更新时间:2015年12月29日13时24分 来源:传智播客Java培训学院 浏览次数:

     前面我们已经介绍了代理的好处了,前面写的是静态代理,要手工创建代理类。下面,我们说说它的问题,我们发现,代理类和老板类都实现了一个接口,如果业务繁多,这样的接口会有很多,如果代理的功能是通用的,就需要对每个接口创建相应的代理类,这个叫类爆炸(类太多了),所以Java提供了动态代理,即程序运行时动态创建代理类的.class.
 
来看一下动态代理:
JDK动态代理中包含一个类和一个接口:
InvocationHandler接口: 代表代理对象关联的处理代码
public interface InvocationHandler {
public Object invoke(Object proxy,Method method,Object[] args) throws Throwable;
}
参数说明:
Object proxy:指被代理的对象(老板)
Method method:要调用的方法
Object[] args:方法调用时所需要的参数
 
调用代理对象的所有方法都会被替换成执行InvocationHandler的invoke()方法
Proxy类:
Proxy类是创建代理对象的工具类,此类提供了如下的操作方法:
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces,
InvocationHandler h) throws IllegalArgumentException
参数说明:
ClassLoader loader:类加载器
Class<?>[] interfaces:得到全部的接口
InvocationHandler h:得到InvocationHandler接口的子类实例 
 
与静态代理类对照的是动态代理类,动态代理类的字节码在程序运行时由Java反射机制动态生成,无需程序员手工编写它的源代码。动态代理类不仅简化了编程工作,而且提高了软件系统的可扩展性,因为Java 反射机制可以生成任意类型的动态代理类。java.lang.reflect 包中的Proxy类和InvocationHandler 接口提供了生成动态代理类的能力。
 
动态代理示例: 
 
//1.演出的接口
public interface Inter{ 
//演出
    public void show();

 
 
//2.某明星(被代理的类,可以理解为老板)
 
public class 明星A  implements Inter { //实现表演接口
 
    //明星表演 
    public void show() { 
        System.out.println("明星A卖力表演中。。。"); 
    } 
 
}  
 
/**
 * 3.JDK动态代理关联的处理代码  
 */ 
public class InvocationHandlerImpl  implements InvocationHandler { 
/*
这个target是经纪人要代理的明星,是Object类型,如果经纪人这个功能是通用的,可以适用于所有要被代理的类.而不用为每种类型的明星都创建一个代理类,这就是动态代理的好处,不像静态代理那样,要为很多老板类创建代理类
*/
private Object target;
 
 
    /**
     * 绑定一个明星并返回一个代理对象(一个经纪人)
     * @param target
     * @return
     */ 
    public Object bind(Object target) { //传入一个要被代理的具体的明星对象
        this.target = target;
        //返回代理对象(这个代理对象实现了和 target一样的接口)
        return Proxy.newProxyInstance(target.getClass().getClassLoader(), 
                target.getClass().getInterfaces(), this);
    } 
 
    @Override 
    /**
     * 调用代理对象的方法都会执行invoke()方法
     */ 
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable  { 
       
        System.out.println("拍戏前签合同");  //拍戏的前期工作
        Object result = method.invoke(target, args);  //中间让明星去表演
        System.out.println("拍戏结束后的后期工作"); //拍戏后的后期工作
        return result; 
    } 
 

 
 
4.客户端调用
 
public class TestProxy { 
 
public static void main(String[] args) { 
//创建关联的处理代码实现类
        InvocationHandlerImpl   handler = new InvocationHandlerImpl();
//返回代理对象(这个代理对象的.class是动态生成的)
        Inter proxy = (Inter) handler .bind(new 明星A());
//调用代理对象的show方法,实际执行的是handler 的invoke()方法
        proxy .show();
    } 
 

 
    可能大家觉得这个动态代理也没什么用?是的,在普通开发中,这个是很少用的,
但在搭建系统框架和底层开发中,是会用到的。
 
    JDK的动态代理依靠接口实现,如果有些类并没有实现接口,则不能使用JDK的动态代理,这就要使用cglib动态代理了。
 
 本文版权归传智播客Java培训学院所有,欢迎转载,转载请注明作者出处。谢谢!
作者:传智播客Java培训学院
首发:http://www.itcast.cn/javaee