Java代理模式详细解析
代理模式是我们比较常用的设计模式之一。其中新思想是为了提供额外的处理或者不同的操作而在实际对象与调用者之间插入一个代理对象。这些额外的操作通常需要与实际对象进行通信,代理模式一般涉及到的角色有:
抽象角色:声明真实对象和代理对象的共同接口;
代理角色:代理对象角色内部含有对真实对象的引用,从而可以操作真实对象,同时代理对象提供与真实对象相同的接口以便在任何时刻都能代替真实对象。同时,代理对象可以在执行真实对象操作时,附加其他的操作,相当于对真实对象进行封装。
真实角色:代理角色所代表的真实对象,是我们最终要引用的对象。
以下以发送消息为例来说明一个简单的代理模式的基本实现:
首先明确目的:有一条消息,需要把这个消息发送出去,根据这个目的定义对应接口messagehandler。需要的附加操作:假设需要验证消息的长度不能超过指定长度并且不能为空,并且我们需要统计相关信息发送到次数,超过指定的次数我们需要输出警报。我们通过代理模式来实现这个附加的操作。下面为对应的类关系图及示例代码。
//接口定义 public interface messagehandler { public void sendmessage(string msg); } //通过email方式发送消息的实现类 public class emailmessage implements messagehandler { @override public void sendmessage(string msg) { // todo auto-generated method stub system.out.println(msg+" send!!"); } } //消息处理的代理类 public class messageproxy implements messagehandler { private static int count; private messagehandler emailmsg; @override public void sendmessage(string msg) { // todo auto-generated method stub if(checkmessage(msg)) { if(emailmsg==null) emailmsg=new emailmessage(); count++; emailmsg.sendmessage(msg); system.out.println("message sent:"+count); } } private boolean checkmessage(string msg) { return msg != null && msg.length() > 10; } } //调用类 public class mainclass { private static void runproxy(messagehandler handler) { handler.sendmessage("message for test"); } /** * @param args */ public static void main(string[] args) { // todo auto-generated method stub runproxy(new emailmessage()); system.out.println("++++++++++++++++pjroxy++++++++++++++++++"); runproxy(new messageproxy()); } } //输出 message for test send!! ++++++++++++++++pjroxy++++++++++++++++++ message for test send!! message sent:1
在例子中我们可以方便的在消息发送过程中添加各种需要的附加处理方式,也能方便的替换消息的处理方式,如将通过email发送消息替换为通过短信发送消息,而调用方不会有丝毫察觉!在任何你想要将一些额外操作分离到具体对象之外,特别是希望能够很容易做出修改,或者想在具体对象的方法执行前插入一些额外操作的时候,代理就显得十分有用!
动态代理
java中动态代理机制的引入使得代理模式的思想更加完善与进步,它允许动态的创建代理并支持对动态的对所代理的方法进行调用。java动态代理类位于java.lang.reflect包下,一般主要涉及到以下两个类:
(1). interface invocationhandler:该接口中仅定义了一个方法object:invoke(object obj,method method, object[] args)。在实际使用时,第一个参数obj一般是指代理类,method是被代理的方法,如上例中的request(),args为该方法的参数数组。这个抽象方法在代理类中动态实现。
(2).proxy:该类即为动态代理类,作用类似于上例中的proxysubject,其中主要包含以下内容:
protected proxy(invocationhandler h):构造函数,估计用于给内部的h赋值。
static class getproxyclass (classloader loader, class[] interfaces):获得一个代理类,其中loader是类装载器,interfaces是真实类所拥有的全部接口的数组。
static object newproxyinstance(classloader loader, class[] interfaces, invocationhandler h):返回代理类的一个实例,返回后的代理类可以当作被代理类使用(可使用被代理类的在subject接口中声明过的方法)。
所谓dynamic proxy是这样一种class:它是在运行时生成的class,在生成它时你必须提供一组interface给它,然后该class就宣称它实现了这些interface。你当然可以把该class的实例当作这些interface中的任何一个来用。当然啦,这个dynamic proxy其实就是一个proxy,它不会替你作实质性的工作,在生成它的实例时你必须提供一个handler,由它接管实际的工作。 下面我们通过动态代理来重新实现上面发送信息的例子!
在上面的例子基础上,我们先添加一个通过短信来发送消息的处理类:
public class smsmessage implements messagehandler { @override public void sendmessage(string msg) { // todo auto-generated method stub system.out.println("sms message :" + msg+" sent !"); } } //动态代理类 import java.lang.reflect.invocationhandler; import java.lang.reflect.method; public class dynamicmessageproxy implements invocationhandler { private static int count; private messagehandler msghandler; public dynamicmessageproxy(messagehandler handler) { msghandler = handler; } @override public object invoke(object proxy, method method, object[] args) throws throwable { // todo auto-generated method stub system.out.println("++++++++=============+++++++++"); system.out.println("proxy:" + proxy.getclass()); system.out.println("method:" + method); system.out.println("++++++++=============+++++++++"); if (args != null && args.length == 1 && checkmessage((string) args[0])) { count++; system.out.println("message sent:" + count); return method.invoke(msghandler, args); } return null; } private boolean checkmessage(string msg) { return msg != null && msg.length() > 10; } } //下面是调用 import java.lang.reflect.proxy; public class mainclass { private static void runproxy(messagehandler handler) { handler.sendmessage("message for test"); } /** * @param args */ public static void main(string[] args) { // todo auto-generated method stub // runproxy(new emailmessage()); // system.out.println("++++++++++++++++proxy++++++++++++++++++"); // runproxy(new messageproxy()); messagehandler handler = new emailmessage(); runproxy(handler); messagehandler proxy = (messagehandler) proxy.newproxyinstance( messagehandler.class.getclassloader(), new class[] { messagehandler.class }, new dynamicmessageproxy( handler)); runproxy(proxy); system.out.println("++++++++++++++++++++++++++++++++++"); // 短信方式 handler = new smsmessage(); runproxy(handler); proxy = (messagehandler) proxy.newproxyinstance(messagehandler.class .getclassloader(), new class[] { messagehandler.class }, new dynamicmessageproxy(handler)); runproxy(proxy); } } //下面为以上方法的输出: message for test send!! ++++++++=============+++++++++ proxy:class $proxy0 method:public abstract void messagehandler.sendmessage(java.lang.string) ++++++++=============+++++++++ message sent:1 message for test send!! ++++++++++++++++++++++++++++++++++ sms message :message for test sent ! ++++++++=============+++++++++ proxy:class $proxy0 method:public abstract void messagehandler.sendmessage(java.lang.string) ++++++++=============+++++++++ message sent:2 sms message :message for test sent !
以上例子中,通过调用proxy.newproxyinstance方法创建动态代理对象,该方法需要传入一个 类加载器、一组希望代理实现的接口列表、invocationhandler 接口的一个具体实现。动态代理可以将所有调用重定向到调用处理器,通常我们会向该处理器传递一个时间对象的引用。invoke()方法中传递进来了代理对象,当你需要区分请求来源时这是非常有用的,例如你可以通过判断传入的方法名屏蔽掉某些方法的执行!动态代理机制并不是会很频繁使用的方法,它通常用来解决一些特定情况下的问题,因此不要盲目的为了使用而使用,要根据自己的实际需求来决定!
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
上一篇: Android 更新UI的方法汇总
下一篇: 亲爱的女儿,生日快乐! 出版生活