Eclipse中的IAdaptable分析

  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。

时间: 2024-09-28 05:11:57

Eclipse中的IAdaptable分析的相关文章

使用Codan在Eclipse中执行C/C++分析工具

Alex Ruiz 解释了为什么该更新对于 http://www.aliyun.com/zixun/aggregation/13428.html">Eclipse CDT 用户来说是一个好消息,然后展示了如何使用 Java 代码和少许 XML 快速将您最喜爱的代码检查器集成到 Eclipse C++/C++ 开发环境中. Codan 是在 C/C++ 项目上执行代码检查的代码分析框架.自 2011 年起,Codan 已成为 Eclipse CDT(C/C++ 开发工具)一部分,它不仅提供执

在Eclipse中创建新的重构功能

创建 对重构的强大支持是软件开发人员喜爱Eclipse的一个最为重要的原因.而Eclipse还有一个至少和重构不相上下的优点,那就是其近乎无懈可击的可扩展性.这两者的结合意味着我们可以根据自己的需要来创建展新的重构功能. 介绍 重构在现代软件开发过程中扮演着重要的角色,它能够减轻软件开发人员的工作负担,提高软件开发的生产效率.为了阐明重构的重要性,我们在这里引用了developerWorks上David Carew提供的关于重构的教程中的一段话: 现在,一个开发者的工作大部分在于对现有的代码进行

在Eclipse中集成Ant编程之配置篇

编程 提要:本文将向你展示如何使用Eclipse设置为Ant所用的属性值和环境变量,并简要分析如何配置Ant编辑器以便从Eclipse内部操作Ant文件. 一. 修改Ant Classpath 在使用一个可选的或定制的任务时,你可能需要把其它库添加到classpath中.为此,你可以通过以全局方式或使用一个单独工程的启动配置来修改Ant classpath.为了给一个单独的Eclipse工程设置Ant classpath,打开该工程的Ant启动配置并点击Classpath选择卡:然后,通过点击"

发现 Eclipse 中未解析的插件依赖性

试图定位无法解析的插件依赖性是件紧张而又耗时的事情.激活每个插件都要依赖于很多其他插件,这些插件又会依赖于其他更多插件.如果 Eclipse 无法加载这个长长的链条中的某个插件,那么手工查找出现问题的到底是哪个插件可能会比原计划所花费的时间和精力都要多.如果您希望有一种方法可以自动实现这种插件依赖性的检测,就请阅读本文. 碰到的问题 假设我们希望在 Eclipse 中使用一个特定的插件,并已经执行了所有必须的操作,将其包含到插件的 manifest 文件中,并将其声明为一个依赖文件.但是系统并没

eclipse中去除build时总报js错误

在用eclipse时经常莫名其名的弹出如下框框,有的时候甚至还死循环了.严重影响开发效率. 原因分析就是我们项目的一些js代码,eclipse验证时有错误的,其实是没有错误的.不知道eclipse是怎么验证的.我靠还是直接取消掉eclipse的验证吧.方法如下 1 window->preference->Java Script->Validator->Errors/Warnings->Enable Javascript Sematic validation 2  找到项目目录

Maven使用笔记(二)Eclipse中maven项目添加依赖

1.在Eclipse中创建Eclipse项目后如何添加jar包? 点击pom.xml文件.我们可以看到下面有7个标签. 各个标签的含义如下: Overview:显示maven项目的一些基本信息Dependencies:添加jar包的页面Plugins:添加maven插件的页面.比如tomcat-maven-plugin等Dependency Hierarchy:用于显示jar包的依赖关系.没事的时候可以看看jar包的依赖关系Effective POM:显示maven的编译路径,plugin之类的

Eclipse中Axis2的配置问题

问题描述 我在Eclipse中使用Axis2的时候,明明在Window-->Preferences-->WebService-->Axis2中配置好了runtimelocation,可是在发布一个service的时候还报出了以下得错误,头疼啊,那位大侠帮忙分析一下原因~Exceptionoccurredwhilereadingorwritingfile{0}TheAxis2facetscannotbeinstalledsincetheAxis2runtimelocationhasnotb

在eclipse中使用maven构建Web项目时报错:maven-resources-plugin 解决方法

今天在一个新建的web项目中发生了以下错误: Failed to read artifact descriptor for org.apache.maven.plugins:maven-resources-plugin Eclipse的错误提示如下: Could not calculate build plan: Plugin org.apache.maven.plugins:maven-resources-plugin:2.6 or one of its dependencies could

Eclipse中jsp、js文件编辑时,卡死现象解决汇总

使用Eclipse编辑jsp.js文件时,经常出现卡死现象,在网上百度了N次,经过N次优化调整后,卡死现象逐步好转,具体那个方法起到作用,不太好讲.将所有用过的方法罗列如下:   1.取消验证 <1>windows–>perferences–>validation 把 除了manual 下面的全部点掉,build下只留 classpath dependency Validator     <2>Window -> Preference -> General