动态代理:
动态代理:本来应该自己做的事情,请了别人来做 被请的人就是代理对象
在Java中要想实现动态代理,需要java.lang.reflect.InvocationHandler接口和 java.lang.reflect.Proxy 类的支持
在Java中java.lang.reflect包下提供了一个Proxy类和一个InvocationHandler接口,通过使用这个类和接口就可以生成动态代理对象。提供的代理只能针对接口做代理。
使用方法介绍:
1)java.lang.reflect.Proxy:
public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)此方法的作用的 返回被代理接口的对象
第一个参数:classLoader loader 类加载器
第二个参数:Class<?> interfaces 得到全部的接口
第三个参数:InvocationHandler 得到InvocationHandler的接口的子类的实例
2)java.lang.reflect.InvocationHandler类的定义如下:
public interface InvocationHandler {
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable; //该invoke方法返回的值是被代理接口的一个实现类
第一个参数:Object proxy:被代理的对象
第二个参数:Method method:要调用的方法 通过反射来获得的method
第三个参数:Object[] args;// 方法调用时所需要的参数
}
动态代理举例(以ArrayList动态代理)
第一种写法(使用了内部类)
动态创建代理对象的类MyInvocationHandler :动态代理类只能代理接口(不支持抽象类),代理类都需要实现InvocationHandler类,实现invoke方法。该invoke方法就是调用被代理接口的所有方法时需要调用的, 该invoke方法返回的值是被代理接口的一个实现类 。
public class MyInvocationHandler implements InvocationHandler{
private Object target;
//构造方法的作用: 我需要用MyInvocationHandler创建子类对象 我需要把被代理的对象传过来
public MyInvocationHandler(Object target){
//接收被代理的对象
this.target=target;
}
/*
* Object proxy:被代理的对象
* Method method:要调用的方法 通过反射来获得的method
* Object[] args;// 方法调用时所需要的参数
*/
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
**在此处可以做一些日志系统、权限控制、拦截器等**
//原对象调用前处理权限校验
System.out.println("权限校验");
//调用目标方法
Object ret =method.invoke(target, args); //执行被代理target对象的方法
//原对象调用后处理日志信息
System.out.println("日志记录");
return ret;
}
}
public static void main(String[] args) {
final List<String> list=new ArrayList<String>();
List<String> list2=(List<String>)Proxy.newProxyInstance(list.getClass().getClassLoader(), list.getClass().getInterfaces(),
//因为改动态代理方法使用的是匿名内部类实现的,所以上面的list对象使用final修饰(被final修饰其地址值不变,才可以在匿名内部类中被使用)
new InvocationHandler(){
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
return method.invoke(list, args);
}
});
list2.add("1111");
System.out.println(list);
}
输出 :权限校验
日志记录
[11111]
第二种:MyInvocationHandler 类跟上边相同 :
public static void main(String[] args) {
List<String> list=new ArrayList<String>();
public static void main(String[] args) {
MyInvocationHandler m1=new MyInvocationHandler(list);
List<String> list2=(List<String>) Proxy.newProxyInstance(list.getClass().getClassLoader(), list.getClass().getInterfaces(), m1);
list2.add("11111");
System.out.println(list);
}