首页 > 开发 > 综合 > 正文

Eclipse中的IAdaptable分析

2024-07-21 02:14:38
字体:
来源:转载
供稿:网友

  java是一种强类型语言,每个实例都必须有指定的类型。实际上,java类型有两种声明类型和 运行时类型 (也可以相应的说是静态类型 和动态类型 ). 像python这样的弱类型语言通常称为无类型,但是这样说并不严谨,因为每个实例都有它的运行时类型。你只是不用事先声明一个实例的类型而已。

  要想调用一个对象中的方法,这个方法需要在声明类型中存在。也就是说,你只能调用定义在父类中的方法,即使该实例是一个确定的子类型:

list list = new arraylist();
list.add("data"); // 在这里没问题
list.ensurecapacity(4); // 这里就不行了ensurecapacity() 只在arraylist中才有。

  如果我们要调用实际类型中的方法,我们首先要将它转为正确的类型。在本例中,我们可以把 arraylist 转为list,因为arraylist实现了list 接口. 也可以在运行时动态的检验,使用 list instanceof arraylist.

  可扩展的接口

  糟糕的是,一个类不能总是实现你所需要实现的接口。可能是因为这只对少数几种情况才有效,或者它是一个没有被关联的库中的类型,或者这个接口在后期又被改变了。

  这种情况就可以使用iadaptable。 你可以把 iadaptable 动态的进行类型转化。使用如下方法避免直接的类型转化:

object o = new arraylist();
list list = (list)o;

  我们可以这样做:

iadaptable adaptable = new arraylist();
list list = (list)adaptable.getadapter(java.util.list.class);

  你可认为它是一种类型动态转化; 我们把adaptable转为list实例。

  为什么不直接转化,而要用额外的getadapter() 呢?这种机制可以使我们将目标类转化为没有实现的接口。例如, 我们可能想使用hashmap 作为一个 list, 尽管他们并不兼容。

iadaptable adaptable = new hashmap();
list list = (list)adaptable.getadapter(java.util.list.class);

  实现iadaptable
 
  大多数iadaptable的实现看起来就想是为支持类型构造多个if表达式的叠加。如果要为hashmap实现getadapter() 可以这样:

public class hashmap implements iadaptable {
 public object getadapter(class clazz) {
  if (clazz == java.util.list.class) {
   list list = new arraylist(this.size());
   list.addall(this.values());
   return list;
  }
  return null;
 }
 // ...
}


  返回的是一个对自身的代理,而不是直接转化类型。如果请求的是不支持的类型,可以直接返回null表明失败,这样比抛出异常要好。

  platformobject

  当你想添加新的要扩展的类型时,只是简单的修改一下就可以了。在任何情况下,如果已经得到了类型,为什么不修改接口?不修改类(如果使用接口,不容易保证向后兼容)或者改变它的类型(hashmap不是 list,但是可以转化)是有原因的。要解决这个问题,在eclipse中,使用了一个抽象类 platformobject。它为你实现了 iadaptable接口,你就可以不用再操心了。

  platformobject 代理所有的它对getadapter()的请求到 iadaptermanager. iadaptermanager是平台默认提供的,通过 platform.getadaptermanager()来访问。你可以将它想象为一个巨大的 map ,它负责关联类和适当的适配器。platformobject的 getadapter() 方法可以访问到这个map.

  适配已存在的类

  这样的好处是可以为每一个platformobject对象动态的关联新的适配器,而不用重新编译。在eclipse中的很多地方都是这样来支持扩展的。

  这里希望将装有string的list转为xml节点。 xml节点显示为:

<list>
<entry>first string</entry>
<entry>second string</entry>
<entry>third string</entry>
</list>

  因为list的tostring方法可能有别的用途,所以不能使用。 可以为list添加一个工厂,当有转为xml节点的请求时,一个node对象就会自动返回。

  这里需要3个步骤:

  1. 从list中生成node

  使用iadapterfactory 来封装转换机制:

import nu.xom.*;
public class nodelistfactory implements iadapterfactory {
 /** the supported types that we can adapt to */
 private static final class[] types = {
  node.class,
 };
 public class[] getadapterlist() {
  return types;
 }
 /** the actual conversion to a node */
 public object getadapter(object list, class clazz) {
  if (clazz == node.class && list instanceof list) {
   element root = new element("list");
   iterator it = list.iterator();
   while(it.hasnext()) {
    element item = new element("entry");
    item.appendchild(it.next().tostring());
    root.appendchild(item);
   }
   return root;
  } else {
   return null;
  }
 }
}

  2. 注册工厂到platform的adaptermanager

  我们需要注册工厂到适配器工厂,当我们向 list实例请求node时, 它就会知道是使用我们注册的工厂。 platform为我们管理iadaptermanager ,而且注册过程相当简单:

platform.getadaptermanager().registeradapters(
new nodelistfactory(), list.class
);

  上面的代码要求平台管理者关联nodelistfactory和list。但我们要求list实例的适配器,它会调用这个工厂。根据我们对工厂的定义,会获得一个node对象。在eclispe中,这一步必须在插件启动的时候显式的执行,要隐式执行可以通过 org.eclipse.core.runtime.adapters 扩展点。

  3. 向list要求node

  这里是要求适配器返回一个 node 对象:

node getnodefrom(iadaptable list) {
 object adaptable = list.getadapter(node.class);
 if (adaptable != null) {
  node node = (node)adaptable;
  return node;
 }
 return null;
}

  总结

  如果你要在运行时为已存在的类添加功能,只要定义一个能完成转换功能的工厂,然后注册工程到 platform的 adaptermanager就可以了. 这项功能可以用来为一个非ui组件注册一个指定的ui组件,同时保持两部分的完全分离。就像在org.rcpapps.rcpnews.ui 和org.rcpapps.rcpnews 插件中的使用。在这些例子中, ipropertysource 在ui插件中,它需要与非ui插件的数据相关联。当ui插件初始化时,它注册ipropertysource 到platform, 当数据对象在浏览器中被选中时,属性视图中就会显示相应的属性。

  很明显, java.util.list不能扩展platformobject, 所以你不能指望例子中的代码能够编译通过,你可以重新构造 list的子类来实现目的.继承platformobject 也不是必须的:

public class adaptablelist implements iadaptable, list {
 public object getadapter(class adapter) {
  return platform.getadaptermanager().getadapter(this, adapter);
 }
 private list delegate = new arraylist();
 public int size() {
  return delegate.size();
 }
 // ...
}

  本例中使用了xom 来生成xml。
发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表