设计模式-动态代理

动态代理
动态代理:本来应该自己做的事情,请了别人来做 被请的人就是代理对象
在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);
		}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值