`

<转>Eclipse插件开发中的Java项目模型

阅读更多

http://developer.51cto.com/art/200906/130101.htm

 

Eclipse插件开发中的Java项目模型是比较重要的内容,也是我们最长接触到的。本文将详细的为大家讲解Java项目模型。

Java项目模型

Eclipse的项目有很多种,包括Java项目、C++项目、C#项目等,每种项目都有自己的特点。我们最常接触到的项目就是Java项目,因此我们重点来讲解一下Java项目模型。

Java模型是用来对与创建、编辑和构建Java程序相关联的对象进行建模的一些类。Java模型类是在org.eclipse.jdt.core中定义的。这些类实现资源的特定于Java的行为,并进一步将Java资源分解成模型元素。

 

Java模型的继承结构图如图3.13所示。

Java模型的继承结构图

图3.13  Java模型的继承结构图

IJavaElement的子类接口还有IMethod、IType等,在这里没有全部列出。Java模型中的类结构比较简单,级次也非常少。

下面介绍一下各个接口的主要方法。

(1)   IJavaElement

exists:判断元素是否存在。处理Java元素与处理资源对象相似。当使用Java元素时,实际上是在使用某些底层的模型对象的句柄。必须使用exists()来确定元素是否真正存在于工作空间中。

getElementName:返回元素的名称。

getJavaModel:返回其对应的JavaModel,返回值类型是IJavaModel。

getJavaProject:返回元素对应的Java工程,返回值类型是IJavaProject。

getParent:返回父元素,返回值类型是IJavaElement。

getResource:返回元素对应的资源,返回值类型是IResource。

(2)   IClassFile

此接口代表编译后的class二进制文件。

isClass:判断是否是Java类。

isInterface:判断是否是接口。

(3)   ICompilationUnit

此接口代表Java源文件。

getAllTypes:返回此文件中定义的所有类型,返回值类型是IType[]。一个Java文件中可以定义多个类型。

getPrimary:返回主类型,返回值类型是ICompilationUnit。

(4)   IJavaModel

此接口表示根 Java 元素,对应于工作空间。是所有具有 Java 性质的项目的父类。它对于Java项目的作用和IWorkspaceRoot对于IProject的作用相似。

contains:判断是否存在指定的资源。

getJavaProject:返回指定名字的Java项目,返回值类型是IJavaProject。

getJavaProjects:返回所有的Java项目,返回值类型是IJavaProject[]。

getWorkspace:返回所在的工作空间。

(5)   IJavaProject

此接口表示Java项目。

IJavaElement findElement(IPath path):返回项目的path路径下的Java元素。

IPackageFragment findPackageFragment(IPath path):返回项目的path路径下的IPackageFragment。

IPackageFragmentRoot findPackageFragmentRoot(IPath path):返回项目的path路径下的IPackageFragmentRoot。

findType:根据一个全名取得此元素的类型,此类有数个重载方法,返回值类型为IType。

getAllPackageFragmentRoots:返回所有的IPackageFragmentRoot,返回值类型是IPackageFragmentRoot[]。

getOutputLocation:返回输出路径,返回值类型是IPath。

getRequiredProjectNames:返回依赖项目,返回值类型是字符串数组。

setOutputLocation:设定输出路径。

(6)   IPackageFragment

此接口表示整个包或者包的一部分。

createCompilationUnit:创建一个ICompilationUnit,返回值类型是ICompilationUnit。

getClassFile:返回指定名称对应的IClassFile,返回值类型是IClassFile。

getClassFiles:返回所有的IClassFile,返回值类型是IClassFile[]。

getCompilationUnit:返回指定名称对应的ICompilationUnit,返回值类型是ICompilationUnit。

getCompilationUnits:返回所有ICompilationUnit,返回值类型是ICompilationUnit[]。

getKind:判断此包是源码包还是普通包,返回值是int型,如等于IPackage- FragmentRoot.K_SOURCE则是源文件包,如等于IPackageFragmentRoot.K_BINARY则为普通包。

hasSubpackages:是否有子包。

(7)   IPackageFragmentRoot

此接口表示一组包段,并将各段映射至底层资源,它可以是文件夹、JAR或ZIP文件。

createPackageFragment:创建一个IPackageFragment,返回值类型是IPackage- Fragment。

getKind:此包段是源码包段还是二进制包段,返回值类型是int,如果等于IPackageFragmentRoot.K_SOURCE则是源文件包段,如果等于IPackageFragment- Root.K_BINARY则为二进制包段。

getPackageFragment:根据包名返回对应的IPackageFragment。

常用工具类

(1)   JavaCore(定义在org.eclipse.jdt.core包下)

JavaCore从Plugin继承,它是JDT插件的生命周期管理器。不过对于第三方插件开发人员来说,它的重要性更多地体现在它提供的一些工具类方法中。

IJavaElement create(IFile file):从文件创建对应的Java元素。

IJavaElement create(IFolder folder):从文件夹创建对应的Java元素。

IJavaProject create(IProject project):得到IProject对应的IJavaProject。

IJavaElement create(IResource resource):从资源创建对应的Java元素。

IJavaModel create(IWorkspaceRoot root):从工作空间根目录得到对应的IJavaModel。

IClassFile createClassFileFrom(IFile file):从文件创建对应的IClassFile。

ICompilationUnit createCompilationUnitFrom(IFile file):从文件创建对应的ICompilationUnit。

style="LINE-HEIGHT: 17.5pt; TEXT-INDENT: 21pt"(2)   JavaUI(定义在org.eclipse.jdt.ui包下)

JavaUI中定义了常用的Java插件界面相关的方法。

createPackageDialog:创建一个包选择对话框,返回值是SelectionDialog。

createTypeDialog:创建一个类型选择对话框,返回值是SelectionDialog。

IEditorPart openInEditor(IJavaElement element):用编辑器打开指定的Java元素并返回编辑器实例。

revealInEditor(IEditorPart part, IJavaElement element):在编辑器中定位元素element。

插件开发中经常会碰到一些常用的技巧,掌握这些技巧可以极大地提高插件的开发效率,并且可以减小插件的体积。下面列出一些常见的技巧。

(1)   由一个普通项目得到Java项目

Java项目是一种特殊的项目,需要注意的是IJavaProject并不是从IProject继承的。不能将一个IProject对象强制转换成一个IJavaProject对象,也不能把一个IJavaProject实例赋值给IProject变量。

由IProject项目得到Java项目的方式:

IJavaProject javaPoject = JavaCore.create(IProject);

由IJavaProject得到IProject的方式:

调用IJavaProject的IProject getProject();

(2)得到工作空间中的所有Java项目

我们可以首先得到工作空间中的所有项目,然后逐个进行转换。不过这不免麻烦了一些,下面介绍更好的方式。IJavaModel是所有Java项目的根,通过它就可以得到所有的Java项目:

  1. IWorkspaceRoot root = ResourcesPlugin.getWorkspace().getRoot();  
  2. IJavaModel jModel = JavaCore.create(root);  
  3. IJavaProject jProject[] = jModel.getJavaProjects(); 

(3)   打开Java编辑器并显示Java编译单元的特定成员

代码如下:

  1. void showMethod(IMember member) {  
  2.     ICompilationUnit cu = member.getCompilationUnit();  
  3.     IEditorPart javaEditor = JavaUI.openInEditor(cu);  
  4.     JavaUI.revealInEditor(javaEditor, member);  

(4)   在工程下创建一个com.cownew包,并创建一个Hello.java文件

 

 

 

 

 

 

 

(5)   打开【打开类型】对话框

以下代码段使用 JavaUI 类来打开【打开类型】对话框:

 

  1. new ProgressMonitorDialog(parent),   
  2. SearchEngine.createWorkspaceScope(), 

用类似方法还可以创建【打开包】和【打开主要类型】对话框。

(6)   打包指定的文件

我们写一些工具的时候也许需要把文件打成jar包,然后进行发布到应用服务器等操作,调用JDT提供的类可简化这个操作(用到的打Jar包的类都在org.eclipse.ui.jarpackager下):

  1. JarPackageData descriptionnew JarPackageData();  
  2. IPath locationnew Path("C:/cownew.jar");  
  3. description.setJarLocation(location);  
  4. description.setSaveManifest(true);  
  5. description.setManifestMainClass(mainType);  
  6. description.setElements(filestoExport);  
  7. IJarExportRunnable runnable=   
  8. description.createJarExportRunnable(parentShell);  
  9. new ProgressMonitorDialog(parentShell).run(true,true, runnable); 

 

 

参数mainType表示Jar包的main类,filestoExport为要打包的文件。

(7)   自动设置Java项目的构建路径

有一些插件会将需要的jar包自动设置到构建路径上,比如使用WTP的新建向导新建web项目的时候就会把web开发需要的jar包自动放入项目的构建路径,使用PDE的“将项目转换为插件项目”功能后项目的构建路径中就增加了插件依赖项的库。那么它们是怎么实现的呢?

Java项目的构建路径有如下几种:源文件夹、二进制库、依赖项目、类路径变量和类路径容器。

 

 

 

 

 

源文件夹

图3.14  源文件夹

构建依赖项目

图3.15  构建依赖项目

Jar和类文件夹依赖

图3.16  Jar和类文件夹依赖

每种不同的构建路径都有不同的作用:源文件夹是把源码进行构建的途径,二进制库是导入少量jar包的方式,依赖项目是供多项目分模块开发使用的,使用类路径变量可以避免二进制包的路径依赖,而类路径容器则为大量二进制库的引入提供了方便。

JDT为这些不同的构建路径提供了一个统一的接口:IClassPathEntry,只要调用IJavaProject的setRawClasspath方法就可以为项目设定构建路径。

 

 

 

 

可以看到setRawClasspath方法需要一个IClasspathEntry数组,数组中的元素就是要设置的每一个构建路径。前面提到的JavaCore类提供了一系列的静态方法来帮助我们生成不同的IClasspathEntry,而无须关注生成的细节。下面来看不同构建路径的添加方式。

①  源文件夹。使用JavaCore.newSourceEntry方法。下面的代码的作用是构造项目MyProject的源文件夹src的类路径条目:

 

JavaCore.newSourceEntry(new Path("/MyProject/src"));

②  二进制库IClasspathEntry。使用JavaCore.newLibraryEntry 方法。下面的代码就是构造MyProject的类文件lib的类路径条目:

 

Path("/MyProject/lib"),null, null,false);  

以下类路径条目具有源代码连接:

 

 

 

 

 

设定关联源代码包有利于代码的跟踪调试。

③    依赖项目。使用JavaCore.newProjectEntry方法。下面的代码就是构造依赖项目MyFramework:

IClassPathEntry prjEntry = JavaCore.newProjectEntry(new

Path("/MyFramework"), true);

④  类路径变量。使用JavaCore.newVariableEntry方法。类路径变量对于整个工作空间来说是全局的,并且可以通过 JavaCore 方法 getClasspathVariable 和 setClasspathVariable 来处理。

可能会注册自动的类路径变量初始化方法,当启动工作空间时,通过扩展点 org.eclipse. jdt.core.classpathVariableInitializer来调用该类路径变量初始化方法。

以下类路径条目指示一个库,该库的位置存放在变量HOME中。使用变量SRC_HOME和SRC_ROOT来定义源代码连接:

IClassPathEntry varEntry = JavaCore.newVariableEntry(

    new Path("HOME/foo.jar"), //库路径

    new Path("SRC_HOME/foo_src.zip"), //源码归档路径

    new Path("SRC_ROOT"), //源码归档根路径

    true);

JavaCore.setClasspathVariable("HOME", new Path("d:/myInstall"), null);

⑤    类路径容器。通过 JavaCore的getClasspathContainer和setClasspathContainer两个方法来处理类路径容器。 

可能会注册一个自动的类路径容器初始化方法,当需要绑定容器时,通过扩展点 org.eclipse.jdt.core.classpathContainerInitializer来被动地调用类路径容器初始化方法。

以下类路径条目指示系统类库容器:

  1. IClassPathEntry varEntry = JavaCore.newContainerEntry(  
  2.                           new Path("JDKLIB/default"),false);  
  3. JavaCore.setClasspathContainer(  
  4. new Path("JDKLIB/default"),  
  5. new IJavaProject[]{ myProject },   
  6.         new IClasspathContainer[] {  
  7.              new IClasspathContainer() {  
  8.                  public IClasspathEntry[] getClasspathEntries() {  
  9.                      return new IClasspathEntry[]{  
  10.                          JavaCore.newLibraryEntry(  
  11. new Path("d:/rt.jar"), null, null, false);  
  12.                      };   
  13.                  }  
  14.                  public String getDescription() {   
  15. return "Basic JDK library container"; }  
  16.                  public int getKind() { return IClasspathContainer.K_SYSTEM; }  
  17.                  public IPath getPath() { return new Path("JDKLIB/basic"); }  
  18.              }             
  19.         },   
  20. null);    

我们只要调用相应的方法创建我们的类路径条目就可以了,然后把这些条目组成的数组通过setRawClasspath方法设定到项目中。需要注意的是如果我们只把要添加的类路径条目传入 setRawClasspath方法的话,就会替换原有的项目构建路径,这常常是我们不希望的。可以调用IJavaProject的 readRawClasspath方法读取项目已有的设置,把我们要设置的构建路径添加到它的后面,然后再调用setRawClasspath方法设定新的项目构建路径。

 

在这个例子中,将要实现一个“为项目添加 lucene支持”的功能,用户在项目上右击,选择菜单中的【为项目添加lucene支持】命令以后,插件就会把lucene的jar包和源码包复制到项目的lib目录下,并且将jar包加入构建路径。如图3.17所示为增加lucene支持前的项目结构。

图3.17  增加lucene支持之前的项目结构

用户在项目上右击,在弹出的快捷菜单中选择【为项目添加lucene支持】命令后的项目结构如图3.18所示。

图3.18  增加lucene支持之后的项目结构

图3.19是项目的构建路径。

图3.19  增加的lucene包

首先新建一个插件工程,并将JDT相关的依赖项加入。然后添加一个org.eclipse.ui.popupMenus的扩展点,如果不熟悉怎么添加,可以使用插件向导中的“弹出菜单”向导。

需要注意contribution的配置,如图3.20所示。

图3.20  contribution的配置

此插件只针对Java项目起作用,因此 objectClass中填入org.eclipse.jdt.core.IJavaProject;adaptable选择true;如果是用向导生成的那么请记住清空nameFilter。下面是核心类ActionAddLucene的实现代码:

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

  1. System.getProperty("file.separator","/");  
  2. JavaCore.newLibraryEntry(project  
  3. .getFullPath(), project  
  4. LUCENESRCJAR).getFullPath(), null,  
  5.  new IClasspathEntry[oldPaths.length + 1];  
  6. LIB + FILESEPARATOR + LUCENEJAR);  
  7. LIB + FILESEPARATOR + LUCENESRCJAR);  
  8. IClasspathEntry entry)  
  9. e.getMessage(), e)); 

下面解释一下代码中的重点部分。

  1. IClasspathEntry[] oldPaths = javaProject.readRawClasspath(); 

读取项目原有的构建路径条目。

  1. IClasspathEntry luceneLibEntry = JavaCore.newLibraryEntry(  
  2. project.getFile(LIB + FILESEPARATOR + LUCENEJAR).getFullPath(),   
  3. project.getFile(LIB + FILESEPARATOR + LUCENESRCJAR).getFullPath(),  
  4. null, false);    

这一句构建lucene的jar包。

第1个参数是二进制jar包的位置,我们的二进制jar包的位置为项目路径下的lib/lucene-1.4.3-src.jar。

第2个参数是jar包对应的源码包的位置。

第3个参数为源码包的根路径,因为有的源码jar包的源码根路径不是jar包的根路径,比如simplejta的源码jar包的格式如图3.21所示。

图3.21  Jar包的结构

对于这种情况就要指定第2个参数为“src”,lucene的源码包的源码根路径就是jar包的根路径,因此我们设置此参数为null。

第4个参数表示是否导出,我们设置为false。

(3)   URL luceneLib = Activator.getDefault().getBundle().getEntry(RESOUCELIB + FILE- SEPARATOR + LUCENEJAR);

我们把“lucene-1.4.3.jar”、 “lucene-1.4.3-src.jar”放到我们插件的“resoucelib”目录下,当用户单击【为项目添加lucene支持】的时候要把这两个文件复制到项目的lib目录下,因此需要首先读取插件路径“resoucelib”目录下的这两个jar包。

读取插件路径下的文件时我们使用插件Activator类提供的方法即可,比如如下调用:

 

就可以读取到插件根目录下的文件“config/my.xml”,返回类型是java.net.URL。

(4)   copyURLToFile(luceneLib, project, LIB + FILESEPARATOR + LUCENEJAR);

Activator.getDefault().getBundle().getEntry读取到的文件位置是URL类型的,我们需要把这个URL对应的文件复制到项目的lib下。下面看一下copyURLToFile的主干代码:

 

 

 

 

 

 

URL类有一个openStream可以打开文件的输入流,IFile也有一个接受输入流的create方法用来创建文件,因此我们只需要把url的输入流输出给IFile的create方法即可。

这里我们也可以由url得到其对应的磁盘上的路径,也可以得到IFile对应的磁盘上的路径,然后使用Java IO来进行文件复制操作。但是这样做不仅代码数量变多了,而且由于使用的不是Eclipse的资源管理API,会带来无法自动刷新等问题,因此建议尽量使用Eclipse提供的API来完成此功能。

 

学习Eclipse插件开发的最好的方式就是研读 Eclipse的源码,而对插件开发者最有参考价值的就是JDT(Java Development Tools)的代码,相信把所有的包研读一遍以后就会成为插件开发的高手了。下面是各个主要包的内容,读者可以根据需要有选择地进行研读。

    org.eclipse.jdt.ui——提供用于在用户界面中显示 Java 元素的支持类。

    org.eclipse.jdt.ui.actions——提供JDT的大部分菜单和按钮的相关代码,为针对Java用户界面行为的应用程序编程接口。

    org.eclipse.jdt.ui.refactoring——重构相关的类。

    org.eclipse.jdt.ui.search——搜索相关的类。

    org.eclipse.jdt.ui.text——提供用于显示 Java 文本的支持类。

    org.eclipse.jdt.ui.text.folding——Java代码编辑器的代码折叠的实现代码。

    org.eclipse.jdt.ui.text.java——Java代码编辑器代码补全的实现代码。

    org.eclipse.jdt.ui.text.java.hover——Java 编辑器中显示文本悬浮的实现代码。

    org.eclipse.jdt.ui.wizards——创建和配置Java元素的向导页。

 

 

 

分享到:
评论

相关推荐

    Eclipse插件开发中的Java项目模型.doc

    Eclipse插件开发中的Java项目模型

    eclipse插件开发之扩展项目特性

    全面的分析讲解eclipse中jdt的模型。使用技巧及注意事项

    Eclipse_UML建模插件开发

    了解uML开发应用,对Eclipse插件体系以及插件开发方法进行了研究,并在此基础上开发出了基于Eclipse平台的UML建模插件。该插件实现了用例图以及静态结构图的绘制。其中,静态结构图包含Java类图以及Java接口图,同时...

    使用EMF开发基于模型的Eclipse插件

    Eclipse Modeling Framework (EMF)是Eclipse 的一个基于Java语言的模型构建框架,它通过提供通用 的模型框架和自动代码生成工具,极大地简化了基于结 构化模型的Eclipse插件或者应用程序的开发

    EMF for Eclipse 4.4插件

    基于Eclipse的模型框架 它是Eclipse MDA(Model Driven Architecture)的一个重要组成部分 是Eclipse中许多项目的基础 e g GEF EMF可以将模型转换成高效的 正确的 和易于定制的Java代码

    Eclipse权威开发指南2.pdf

    1.1 Eclipse作为应用程序开发 工具的宿主...... 4 1.2 不仅仅是工具的宿主,Eclipse 还作为客户端应用程序的 宿主...... 5 1.3 什么是Eclipse?...... 6 1.3.1 工具与应用程序的集成平台..... 6 1.3.2 Java开发...

    eclipse中的uml类图插件jar包

    org.eclipse.uml2.uml:这是Eclipse UML插件的核心库,提供了UML类图的基本元素,如类、接口、关联、活动图等。它还包含了用于创建和编辑UML模型的编辑器。 org.eclipse.uml2.uml.diagram:这个库包含了用于创建和...

    Eclipse插件开发-测试用例自动生成工具.zip

    开发工具在软件开发生命周期中扮演着至关重要的角色,它们旨在简化和加速从概念设计到产品部署的各个环节。以下是开发工具的主要作用: 代码编写与编辑: 提供集成开发环境(IDE),如Visual Studio、Eclipse、...

    JAVA上百实例源码以及开源项目源代码

     用JAVA开发的一个小型的目录监视系统,系统会每5秒自动扫描一次需要监视的目录,可以用来监视目录中文件大小及文件增减数目的变化。 Java日期选择控件完整源代码 14个目标文件 内容索引:JAVA源码,系统相关,日历,...

    Eclipse权威开发指南1.pdf

    1.1 Eclipse作为应用程序开发 工具的宿主...... 4 1.2 不仅仅是工具的宿主,Eclipse 还作为客户端应用程序的 宿主...... 5 1.3 什么是Eclipse?...... 6 1.3.1 工具与应用程序的集成平台..... 6 1.3.2...

    eclipse导出UML类图插件

    ModelGoon 是一个 Eclipse插件,能将Eclipse中现有的java类生成类图,可以进行Java 包的依赖分析,基于UML图进行模型设计,以及逆向工程(即从已有源代码生成类图)。

    eclipse 开发c/c++

    这些插件将 C/C++ 透视图添加到 Eclipse 工作台(Workbench)中, 现在后者可以用许多视图和向导以及高级编辑和调试支持来支持 C/C++ 开发。 由于其复杂性,CDT 被分成几个组件,它们都采用独立插件的形式。 每个...

    插件开发简介

    Eclipse插件开发简介 SWT概述 SWT/JFace的事件模型 常用插件扩展点

    Java开发热门项目

    Spring是一个解决了许多在J2EE开发中常见的问题的强大框架。 Spring提供了管理业务对象的一致方法并且鼓励了注入对接口编程而不是对类编程的良好习惯。...【SpringIDE:Eclipse平台下一个辅助开发插件】.

    Eclipse权威开发指南3.pdf

    1.1 Eclipse作为应用程序开发 工具的宿主...... 4 1.2 不仅仅是工具的宿主,Eclipse 还作为客户端应用程序的 宿主...... 5 1.3 什么是Eclipse?...... 6 1.3.1 工具与应用程序的集成平台..... 6 1.3.2...

    EasyMda:一个Eclipse插件,可从基于Java的模型生成源代码-开源

    easyMda是一个Eclipse插件,可以从基于Java类的模型生成源代码。 这个项目的主要目标是提供:最灵活(可以想象的任何东西,可以建模和生成),最容易开发和使用,并且非常快速:MDA /代码生成器工具。 与其他Mda工具...

    JAVA上百实例源码以及开源项目

     用JAVA开发的一个小型的目录监视系统,系统会每5秒自动扫描一次需要监视的目录,可以用来监视目录中文件大小及文件增减数目的变化。 Java日期选择控件完整源代码 14个目标文件 内容索引:JAVA源码,系统相关,日历,...

    eclipse-SDK-4.30-win32-x86-64.zip

    Eclipse是一个开放源代码的、基于Java的可扩展开发平台。它本身是一个框架和一组服务,用于通过插件组件构建开发环境。...总的来说,Eclipse是一个免费且功能强大的软件开发工具,广泛应用于各种Java开发项目中。

    java版按键精灵源码-rhostudio:用于开发Rhodes和RhoConnect应用程序的Eclipse插件

    插件,用于促进移动应用程序的开发。 使用 RhoStudio 来: 生成 Rhodes 应用程序 生成 Rhodes 模型和关联的控制器和视图模板 使用 Eclipse UI 管理构建配置(build.yml 和 rhobuild.yml) 在 iPhone、Android、...

    Android开发案例驱动教程 配套代码

    3.1.1 在Eclipse中创建项目 15 3.1.2 编写程序项目代码 17 3.1.3 运行HelloAndroid 18 3.1.4 Android工程目录 19 3.1.5 AndroidManifest.xml文件 21 3.2 Android中的组件介绍 22 3.3 使用Android SDK帮助 23 ...

Global site tag (gtag.js) - Google Analytics