java设计模式之代理模式【设计模式】

2020-04-18 00:33:43 浏览数 (2)

先讲一个段子,有四个人物,分别是张三,我,张三老婆,日本某公司(自己脑补哈)

静态代理

张三想要去日本某公司买xxx,但是对于经费等等一系列的原因然后就放弃了这个念头,我刚好要去日本玩的,张三得知我要去日本,他就偷偷给我说,他想要一个size为D的xxx,让我帮他代购一个,于是我就带着他的需求去日本某公司购入了xxx,然后买完之后回国拿给了张三。 在这个过程中,我作为代理对象(我带着张三的需求【购买的动作即为接口,他的需求即为方法和参数】),张三作为被代理对象。我帮他购买这个过程就叫做代理。 又有一个环境,即当张三的老婆也需要女性的xxx,也偷偷的让我代购,那么我又要亲自过去帮她买,但是想想,我作为一个d代理,一会儿帮这个买,一会儿帮那个买,每次买的东西还不一样,想想,如果作为代码写成一个系统代码是不是会非常的多?

动态代理

我开了一家代购公司,张三联系到我们公司,我们公司的A员工刚好要去日本做代购,于是带着他的需求去代购了,张三老婆也要买日本的另外一款产品,B员工帮他去代购了,这样的代理对象永远都是动态的(Proxy.newInstance(classLoader,classes,invocationHandler)),并且一旦根据被代理对象的需求买完产品之后(method.invoke即执行被代理对象的目标方法)带回过国。 接下来用代码实现以下动态代理:

首先,先创建一个interface

IHello 目标接口类

代码语言:javascript复制
interface IManFanctory {
        Object buy(int size);
 }

然后再写一个目标类的实现类

代码语言:javascript复制
class ManFanctoryImpl implements  IManFanctory {
     @Override
        public  Object buy(int size) {
            System.out.println("buying!");
			return "size:"   size;
        }
 }

最后再手动编写一个代理类

代码语言:javascript复制
//需要实现InvocationHandler接口完成代理
class ManProxy implements InvocationHandler {
 //目标接口 目的拿到目标接口的所有信息
	private IManFanctory  hello; 
	public ManProxy(IManFanctory hello) {
		this.hello = hello;
	}  
	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
	   //利用反射执行该方法  参数1 目标类,参数2  目标类的参数
		return method.invoke(hello, args);
	}
}

然后写一个客户端,对该代理进行操作:

代码语言:javascript复制
@org.junit.Test
public void test() {
	//目标类的类加载器  目标类实现的接口  代理类
	IManFanctory  hello = new ManFanctoryImpl();
	//调用jdk代理方法 实现对classLoader的操作,完成代理,最后使用代理对该方法进行代理操作
	IHello tho = (IHello)Proxy.newProxyInstance(hello.getClass().getClassLoader(), hello.getClass().getInterfaces(),
			(proxy, method, args) -> {
				return method.invoke(hello, args);
			});
	tho.buy(36);
}

0 人点赞