`
zhaohaolin
  • 浏览: 1017564 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

用Java动态代理实现委托模式

    博客分类:
  • JAVA
阅读更多

用Java动态代理实现委托模式

委托模式 是软件设计模式中的一项基本技巧。在委托模式中,有两个对象参与处理同一个请求,接受请求的对象将请求委托给 另一个对象来处理。委托模式是一项基本技巧,许多其他的模式,如状态模式、策略模式、访问者模式本质上是在更特殊的场合采用了委托模式。委托模式使得我们 可以用聚合来替代继承,它还使我们可以模拟mixin。
“委托”在C#中是一个语言级特性,而在Java语言中没有直接的对应,但是我们可以通过动态代理来实现委托!代码如下:

Java代码  收藏代码
  1. import  java.lang.reflect.InvocationHandler;  
  2. import  java.lang.reflect.Method;  
  3. import  java.lang.reflect.Proxy;  
  4. /*  
  5.  * @author Liusheng  
  6.  * 实现“委托”模式,用户需要实现InvocationHandler接口;  
  7.  * 参考:http://www.uml.org.cn/j2ee/200411036.htm  
  8.  */   
  9. public   abstract   class  Delegator  implements  InvocationHandler {  
  10.     //RelegateTo针对每个对象都要生成一个实例,因而非Static的log,代价比较高。   
  11.     //protected Log _log = LogFactory.getLog(this.getClass());     
  12.     //private static Log _log = LogFactory.getLog(RelegateTo.class);   
  13.     //--------------------------------------------   
  14.     protected  Object obj_orgin =  null ;   //原始对象   
  15.     protected  Object obj_proxy =  null ;   //代理对象   
  16.     //--------------------------------------------   
  17.     public  Delegator()  {  
  18.         //空   
  19.     }  
  20.     public  Delegator(Object orgin){  
  21.         this .createProxy(orgin);  
  22.     }  
  23.     //--------------------------------------------   
  24.     protected  Object createProxy(Object orgin) {  
  25.         obj_orgin = orgin;  
  26.         obj_proxy = Proxy.newProxyInstance(  
  27.                 orgin.getClass().getClassLoader(),  //加载器   
  28.                 orgin.getClass().getInterfaces(),   //接口集   
  29.                 this );   //委托   
  30.         //_log.debug("# 委托代理:"+obj_proxy);   
  31.         return  obj_proxy;  
  32.     }     
  33.     protected  Object invokeSuper(Method method, Object[] args)  
  34.     throws  Throwable {  
  35.         return  method.invoke(obj_orgin, args);    
  36.     }  
  37.     //--------------实现InvocationHandler接口,要求覆盖------------   
  38.     public  Object invoke(Object obj, Method method, Object[] args)  
  39.     throws  Throwable {  
  40.         // 缺省实现:委托给obj_orgin完成对应的操作   
  41.         if  (method.getName().equals( "toString" )) {   //对其做额外处理   
  42.             return   this .invokeSuper(method, args)+ "$Proxy" ;  
  43.         }else  {      //注意,调用原始对象的方法,而不是代理的(obj==obj_proxy)   
  44.             return   this .invokeSuper(method, args);  
  45.         }  
  46.     }  
  47. }  
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/*
 * @author Liusheng
 * 实现“委托”模式,用户需要实现InvocationHandler接口;
 * 参考:http://www.uml.org.cn/j2ee/200411036.htm
 */
public abstract class Delegator implements InvocationHandler {
	//RelegateTo针对每个对象都要生成一个实例,因而非Static的log,代价比较高。
	//protected Log _log = LogFactory.getLog(this.getClass());	
	//private static Log _log = LogFactory.getLog(RelegateTo.class);
	//--------------------------------------------
	protected Object obj_orgin = null;	//原始对象
	protected Object obj_proxy = null;	//代理对象
	//--------------------------------------------
	public Delegator()	{
		//空
	}
	public Delegator(Object orgin){
		this.createProxy(orgin);
	}
	//--------------------------------------------
	protected Object createProxy(Object orgin) {
		obj_orgin = orgin;
		obj_proxy = Proxy.newProxyInstance(
				orgin.getClass().getClassLoader(),	//加载器
				orgin.getClass().getInterfaces(),	//接口集
				this);	//委托
		//_log.debug("# 委托代理:"+obj_proxy);
		return obj_proxy;
	}	
	protected Object invokeSuper(Method method, Object[] args)
	throws Throwable {
		return method.invoke(obj_orgin, args);	
	}
	//--------------实现InvocationHandler接口,要求覆盖------------
	public Object invoke(Object obj, Method method, Object[] args)
	throws Throwable {
		// 缺省实现:委托给obj_orgin完成对应的操作
		if (method.getName().equals("toString")) {	//对其做额外处理
			return this.invokeSuper(method, args)+"$Proxy";
		}else {		//注意,调用原始对象的方法,而不是代理的(obj==obj_proxy)
			return this.invokeSuper(method, args);
		}
	}
}


下面的代码,则是作为一个委托的例子,实现Map的功能。

Java代码  收藏代码
  1. import  java.io.IOException;  
  2. import  java.lang.reflect.Method;  
  3. import  java.util.Hashtable;  
  4. import  java.util.Map;  
  5. import  org.apache.commons.logging.Log;  
  6. import  org.apache.commons.logging.LogFactory;  
  7. import  com.bs2.core.UtilLog;  
  8. /**  
  9.  * @author Liusheng  
  10.  * 本代码主要用于演示RelegateTo的使用方法  
  11.  */   
  12. public   class  Delegator4Map  extends  Delegator {  
  13.     private   static  Log _log = LogFactory.getLog(Delegator4Map. class );  
  14.     private  Map orginClass =  null ;   //原始对象   
  15.     private  Map proxyClass =  null ;   //代理对象   
  16.       
  17.     public  Map getOrgin() {  return  orginClass;  }  
  18.     public  Map getProxy() {  return  proxyClass;  }  
  19.       
  20.     public  Delegator4Map(Map orgin) {  
  21.         super (orgin);  
  22.         orginClass = orgin;  
  23.         proxyClass = (Map)super .obj_proxy;  
  24.     }  
  25.     public  Object invoke(Object obj, Method method, Object[] args)  
  26.     throws  Throwable {  
  27.         if  (method.getName().equals( "size" )) {   //修改close处理逻辑   
  28.             _log.debug("原始 size()=" + super .invoke(obj, method, args));  
  29.             Object res2 = new  Integer(- 1 );  
  30.             _log.debug("修改 size()=" +res2);  
  31.             return  res2;  
  32.         }else  {  
  33.             return   super .invoke(obj, method, args);  
  34.         }  
  35.     }     
  36.     public   static   void  main(String[] args)  throws  IOException {  
  37.         UtilLog.configureClassPath("resources/log4j.properties" false );  
  38.         Delegator4Map rtm = new  Delegator4Map( new  Hashtable());  
  39.         Map m = rtm.getProxy();  
  40.         m.size();  
  41.         _log.debug("代理:" +m.toString());  
  42.     }  
  43. }  
import java.io.IOException;
import java.lang.reflect.Method;
import java.util.Hashtable;
import java.util.Map;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import com.bs2.core.UtilLog;
/**
 * @author Liusheng
 * 本代码主要用于演示RelegateTo的使用方法
 */
public class Delegator4Map extends Delegator {
	private static Log _log = LogFactory.getLog(Delegator4Map.class);
	private Map orginClass = null;	//原始对象
	private Map proxyClass = null;	//代理对象
	
	public Map getOrgin() { return orginClass;	}
	public Map getProxy() { return proxyClass;	}
	
	public Delegator4Map(Map orgin) {
		super(orgin);
		orginClass = orgin;
		proxyClass = (Map)super.obj_proxy;
	}
	public Object invoke(Object obj, Method method, Object[] args)
	throws Throwable {
		if (method.getName().equals("size")) {	//修改close处理逻辑
			_log.debug("原始 size()="+super.invoke(obj, method, args));
			Object res2 = new Integer(-1);
			_log.debug("修改 size()="+res2);
			return res2;
		}else {
			return super.invoke(obj, method, args);
		}
	}	
	public static void main(String[] args) throws IOException {
		UtilLog.configureClassPath("resources/log4j.properties", false);
		Delegator4Map rtm = new Delegator4Map(new Hashtable());
		Map m = rtm.getProxy();
		m.size();
		_log.debug("代理:"+m.toString());
	}
}


注意:UtilLog仅仅是用于配置log4j属性文件位置,如果log4j.properties就在缺省的运行路径下,则无需单独配置。或者用System.out输出来替代_log输出。

分享到:
评论

相关推荐

    基于Java动态代理的AOP实现.zip

    本项目旨在通过Java的动态代理技术实现面向切面编程(AOP),涵盖了静态代理、JDK动态代理、CGLIB动态代理、AspectJ以及基于instrumentation的动态代理等多种实现方式。通过这些实现方式,可以在不修改原有代码的...

    java动态代理机制分析及拓展

    在Java中,代理类和委托类通常实现相同的一组接口,这样代理类就可以代替委托类被客户端调用,而客户端无须知道实际处理请求的对象是代理还是委托。 **Proxy类**: 1. `getInvocationHandler`: 获取指定代理对象...

    Java静态代理和动态代理

    Java的代理模式通过代理类提供了对委托类的扩展和控制,静态代理适合对已有代码不做修改的情况,而动态代理则提供了更高的灵活性和扩展性。在实际应用中,应根据项目需求和性能考虑选择静态代理或动态代理。对于需要...

    java动态代理机制综合分析以及实现

    4. **使用动态代理实例**:通过动态代理实例调用相应的方法,这些方法会被代理类转发到委托类进行实际执行。 #### 动态代理类的生成与实现 当调用`Proxy.newProxyInstance`方法时,Java虚拟机会根据传入的`...

    观察者模式Vs事件委托Demo

    在Java中,`java.util.Observable` 类和 `java.util.Observer` 接口提供了一个实现观察者模式的基础框架。当被观察的对象(Subject)状态改变时,它会调用 `notifyObservers()` 方法,将变化通知给所有注册的观察者...

    java动态代理(JDK和cglib).pdf

    Java动态代理是一种强大的设计模式,它允许我们在运行时创建具有特定行为的代理对象。这种模式在处理如AOP(面向切面编程)、事件监听、事务管理等场景中非常有用。动态代理主要有两种实现方式:JDK动态代理和CGLIB...

    Java动态代理机制综合分析及

    这一机制使得开发者无需手动编写代理类代码,只需指定一组接口和目标对象(委托类),就能动态生成实现了这些接口的代理类实例。 Proxy类是Java动态代理的核心,它提供了以下几个关键的静态方法: 1. `...

    java静态代理与动态代理

    无论是静态代理还是动态代理,它们都是代理模式的具体实现形式。选择哪种方式取决于具体的应用场景和技术需求。静态代理适用于功能较为固定的场景,而动态代理则更适合于需要高度灵活性和可扩展性的应用。理解这两种...

    java静态代理和动态代理详解

    Java中的代理模式是一种设计模式,它允许我们创建一个代理对象,该对象可以在调用实际对象的方法之前或之后执行额外的操作,而无需修改原始代码。在Java中,代理主要分为静态代理和动态代理。 1. 静态代理: 静态...

    java 使用委托控制权限

    Java代理机制允许我们在运行时动态创建一个实现了一组给定接口的新类,这个新类可以作为原类的代理,提供额外的功能,如日志、事务管理或者,如题目中所述,权限控制。JDK代理是Java标准库提供的一个实现,位于`java...

    Java动态代理[参考].pdf

    Java动态代理机制是一种在运行时创建代理类和代理对象的技术,它允许我们为已存在的接口创建代理实现,以便在调用接口方法时添加额外的功能或行为。这种模式在软件开发中非常有用,尤其是在需要扩展已有功能,比如...

    JAVA代理

    总结一下,`JAVA代理`涉及到的主要知识点包括: 1. **静态代理**:手动创建代理类,通过代理类对目标对象进行方法调用的拦截和增强。 2. **动态代理**:利用Java的`Proxy`类和`InvocationHandler`接口,动态地创建...

    JAVA23中设计模式

    ### JAVA23中设计模式详解 #### 一、概述 设计模式是在软件设计过程中解决常见问题的一套可复用的解决方案。《JAVA23中设计模式》是一份针对Java程序员的指南,旨在通过一系列示例和理论讲解,帮助读者理解和掌握...

    Spring AOP的静态代理和动态代理,转移到调用处理器一个集中的方法中处理.docx

    在Java中,我们可以使用静态代理和动态代理来实现这一目的。 **静态代理** 静态代理是在编译时就确定了代理类的结构,代理类的源代码是程序员手动编写或者由特定工具自动生成的。代理类需要实现与目标类相同的接口...

    Java JDK代理、CGLIB、AspectJ代理分析比较

    ### Java代理技术详解:JDK代理、CGLIB与AspectJ #### 一、代理模式简介 代理模式是一种常见的设计模式,在《Design Patterns in Java》一书中对其有明确的定义:“代理模式为一个对象提供一个代理,以控制对该...

    动态代理例子

    在提供的`ProxyTest`文件中,可能包含了这样的示例代码,演示了如何创建和使用动态代理。你可以通过查看和运行这段代码来加深对动态代理的理解。 总结来说,动态代理是Java中一种强大的设计模式,它允许我们为现有...

    Java理论与实践:用动态代理进行修饰

    总的来说,Java的动态代理机制提供了一种动态实现设计模式和增强对象功能的方法,减少了代码冗余,提高了代码的可维护性和可扩展性。它是一个强大且未充分利用的工具,值得开发者深入了解和熟练掌握。

Global site tag (gtag.js) - Google Analytics