Java的委托

24 篇文章 0 订阅

委托一般称为委托调用。

代码编写原则:

能不用继承就不用继承,能使用委托实现的就不使用继承。两个类有明显示的层级关系时使用继承,没有明显的层级关系,仅仅是为了在一个类中使用另一个类的方法时应该使用委托。

举例:
根据《重构》一书称:现在有滥用继承的趋势,JDK 中 Stack 就是一个滥用继承的典型!
java.util.Stack 继承自 java.util.Vector,其实 Stack 与 Vector 在用途上完全是风马牛不相及的两个容器。 Stack 和 Vector 的功能是不一样的,Stack 是不允许插入数据到 Stack 任何位置的,但是 Vector 是可以的,如果 Stack 继承自 Vector,那么这样的 Stack 就可以使用 add(int index, E element) 在任何位置插入数据,这明显不符合栈的要求。


java中没有直接可以用的委托,但可以通过动态代理来实现。


在委托模式中,有两个对象参与处理同一个请求,接受请求的对象将请求委托给另一个对象来处理。

委托模式使得我们可以用聚合来替代继承,它还使我们可以模拟mixin。

“委托”在C#中是一个语言级特性,而在Java语言中没有直接的对应,但是我们可以通过动态代理来实现委托!代码如下:

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
    
 public abstract class Delegator implements InvocationHandler {
     //--------------------------------------------

    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;
         //下面语句中orgin.getClass().getClassLoader()为加载器,orgin.getClass().getInterfaces()为接口集
        obj_proxy = Proxy.newProxyInstance(orgin.getClass().getClassLoader(), orgin.getClass().getInterfaces(), this); //委托
        return obj_proxy;
     }
    

     protected Object invokeSuper(Method method, Object[] args) throws Throwable {
         return method.invoke(obj_orgin, args);
     }
     //--------------实现InvocationHandler接口,要求覆盖------------
    //下面实现的方法是当委托的类调用toString()方法时,操作其他方法而不是该类默认的toString(),这个类的其他方法则不会。

    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的功能。

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;

 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")) { //修改size处理逻辑
            Object res2 = new Integer(-1);
             System.out.println("调用委托的方法");
             return res2;
         } else {
             System.out.println("调用原始的方法");
             return super.invoke(obj, method, args);
         }
     }

     public static void main(String[] args) throws IOException {
         Delegator4Map rtm = new Delegator4Map(new Hashtable());
         Map m = rtm.getProxy();
         m.size();
     }
 }




  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值