前面我们已经介绍了代理的好处了,前面写的是静态代理,要手工创建代理类。下面,我们说说它的问题,我们发现,代理类和老板类都实现了一个接口,如果业务繁多,这样的接口会有很多,如果代理的功能是通用的,就需要对每个接口创建相应的代理类,这个叫类爆炸(类太多了),所以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 接口提供了生成动态代理类的能力。
动态代理示例:
public interface Inter{
public void show();
}
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;
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();
Inter proxy = (Inter) handler .bind(new 明星A());
proxy .show();
}
}
可能大家觉得这个动态代理也没什么用?是的,在普通开发中,这个是很少用的,
但在搭建系统框架和底层开发中,是会用到的。
JDK的动态代理依靠接口实现,如果有些类并没有实现接口,则不能使用JDK的动态代理,这就要使用cglib动态代理了。