【设计模式】代理模式Proxy---动态代理

上一篇说到的静态代理,但是有新的接口,就需要新的代理,那么我们需要一个动态代理。java.lang.reflect包里,有个封装类Proxy,正好解决这个问题。

有很多很多的人都说过Proxy,无非就是这行代码的作用,Proxy.newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)根据参数(接口),以反射机制生成动态代理,执行相应接口方法。

1、参数(接口)

/**
 * 公共接口
 * @author Biligle
 *
 */
public interface IPublicManager {
	/**
	 * 排水
	 */
	void out();
	/**
	 * 发电
	 */
	void power();
	/**
	 * 净化
	 */
	void clean(String params);
}


2、反射机制生成动态代理,执行相应接口方法。
/**
 * 生成动态代理,执行接口方法
 * @author Biligle
 *
 */
public class ProxyUtil {
	
	public <T> Object create(Class<T> service){
		
		if(!service.isInterface()){
			new IllegalArgumentException("API声明必须是接口");
		}else if(service.getInterfaces().length > 0){
			new IllegalArgumentException("API接口不能扩展其他接口");
		}
		/**
		 * ClassLoader loader:类加载器 
          Class<?>[] interfaces:得到全部的接口 
          InvocationHandler h:
		 */
		return Proxy.newProxyInstance(service.getClassLoader(), new Class[]{service}, new InvocationHandler() {
			
			/**
			 * proxy 代理对象
			 * method 接口方法
			 * args 接口方法的参数
			 */
			@Override
			public Object invoke(Object proxy, Method method, Object[] args)
					throws Throwable {
				//无论代理去执行哪个method,首先统一执行一个方法common,之后,在分别执行各自方法
				//简言之,这里可以添加公共方法
				BasicVoid.common(method,args);
				return null;
			}
		});
	}
}
/**
 * 代理模式统一执行方法
 * @author Biligle
 *
 */
public class BasicVoid {
	static void common(Method method, Object[] args){
		System.out.println("接收到" + method.getName() + "接口" + "+parmas" + ":" + args[0]);
		System.out.println("给" + method.getName() + "接口方法返回相应数据");
	}
}
测试

public class Test {
	public static void main(String[] args) {
		ProxyUtil p = new ProxyUtil();
		IPublicManager iPublic = (IPublicManager) p.create(IPublicManager.class);
		iPublic.clean("纯净水");
	}
}

测试结果:

接收到clean接口+parmas:纯净水
给clean接口方法返回相应数据


扫描二维码关注公众号,回复: 1687242 查看本文章

大家应该发现了,我们没有去实现clean方法,而是用BasicVoid.common(method,args)做了统一处理,这样的好处是,我们又少写了一个实现类。

下面,我们写一下处理逻辑,这里需要一个接口,来给clean返回一些相应数据。

对BasicVoid修改如下:

/**
 * 代理模式统一执行方法
 * @author Biligle
 *
 */
public class BasicVoid {
	
	private BasicVoid(){}
	private static VisitListener listener;
	private static Object data;
	public static BasicVoid basicVoid = null;
	
	public static BasicVoid get(){
		if(basicVoid == null){
			basicVoid = new BasicVoid();
		}
		return basicVoid;
	}
	
	public BasicVoid common(Method method, Object[] args){
		data = visit(method.getName());
		return basicVoid;
	}
	
	/**
	 * 传入接口,用于监听返回数据
	 * @param l
	 * @return
	 */
	public BasicVoid call(VisitListener l){
		listener = l;
		listener.callData(data);
		return basicVoid;
	}
	
	/**
	 * 统一标准访问,并返回相应数据
	 * @param method
	 * @return
	 */
	private Object visit(String method){
		return method;
	}
	
	/**
	 * 获取数据的监听
	 */
	interface VisitListener {
		/**
		 * 回调数据
		 * @param obj
		 */
		void callData(Object obj);
	}
}
对ProxyUtil修改如下:

/**
 * 生成动态代理,执行接口方法
 * @author Biligle
 *
 */
public class ProxyUtil implements BasicVoid.VisitListener {
	
	public <T> Object create(Class<T> service){
		
		if(!service.isInterface()){
			new IllegalArgumentException("API声明必须是接口");
		}else if(service.getInterfaces().length > 0){
			new IllegalArgumentException("API接口不能扩展其他接口");
		}
		/**
		 * ClassLoader loader:类加载器 
          Class<?>[] interfaces:得到全部的接口 
          InvocationHandler h:
		 */
		return Proxy.newProxyInstance(service.getClassLoader(), new Class[]{service}, new InvocationHandler() {
			
			/**
			 * proxy 代理对象
			 * method 接口方法
			 * args 接口方法的参数
			 */
			@Override
			public Object invoke(Object proxy, Method method, Object[] args)
					throws Throwable {
				//无论代理去执行哪个method,首先统一执行一个方法common,之后,在分别执行各自方法
				//简言之,这里可以添加公共方法
				BasicVoid.get().common(method, args).call(ProxyUtil.this);
				return null;
			}
		});
	}

	@Override
	public void callData(Object obj) {
		System.out.println("返给了" + obj);
	}
}

测试结果

返给了clean

好了,最后在强调一下代理模式的核心概念

根据参数(接口IPublicManager.class),以反射机制生成动态代理,执行相应接口方法(clean)。






猜你喜欢

转载自blog.csdn.net/qq_29266921/article/details/73294597