equinox环境下开发bundle不需要引入java.*包而需要引入javax.*包的的原因
一、前提知识 ClassLoader
ClassLoader(类加载器)是Java提供的抽象类,它是负责加载类的对象。ClassLoader 做的工作就是在JVM 中将类装入内存。当 JVM 需要使用类时,它根据名称向 ClassLoader 请求这个类,然后 ClassLoader 返回一个表示这个类的 Class 对象。
所有java.*包内的类都是由ClassLoader完成加载的,而且在执行java命令的时候ClassLoaer就已经被装入内存中。通常在开发过程中使用java.*包内的类的时候,不再需要import 这些类了。
二、Bundle中的ClassLoader
在equinox环境下,每一个bundle都有独立的classLoader,对应的类名为BundleLoader。
每个bundle的classLoader都有parent,这个parent(父classLoader)就是类的委派模型中的上一层的classLoader。对应的类名为ParentClassLoader,它是这样定义的
privatestaticclassParentClassLoaderextendsClassLoader{
protectedParentClassLoader(){
super(null);
}
}
这是一个空的classloader,它直接继承了java.lang.ClassLoader 。java包中的所有类都是由java.lang.ClassLoader载入的,根据类的委派模型可以知道,继承了ClassLoader就可以直接获得需要的java类。
if(name.startsWith(JAVA_PACKAGE))
//1)ifstartsWith"java."delegatetoparentandterminatesearch
//wewanttothrowClassNotFoundExceptionsifajava.*classcannotbeloadedfromtheparent.
returnparent.loadClass(name);
上面代码是bundle加载class的流程图中的第一步,通过parentClassLoader获得所需java.*包中的类,如下图所示。
这就是bundle中不需要引入java.*包的原因。
三、如何知道java.*这些类已经被装载了呢?
1、写一个类如下:
publicclassA
{
publicstaticvoidmain(String[]args)
{
System.out.println("HelloWorld!");
}
}
2、编译
javacA.java
3、运行
java-verboseA
显示的部分内容如下:
[Loadedjava.lang.Objectfromsharedobjectsfile]
[Loadedjava.io.Serializablefromsharedobjectsfile]
[Loadedjava.lang.Comparablefromsharedobjectsfile]
[Loadedjava.lang.CharSequencefromsharedobjectsfile]
[Loadedjava.lang.Stringfromsharedobjectsfile]
[Loadedjava.lang.reflect.GenericDeclarationfromsharedobjectsfile]
[Loadedjava.lang.reflect.Typefromsharedobjectsfile]
[Loadedjava.lang.reflect.AnnotatedElementfromsharedobjectsfile]
[Loadedjava.lang.Classfromsharedobjectsfile]
[Loadedjava.lang.Cloneablefromsharedobjectsfile]
[Loadedjava.lang.ClassLoaderfromsharedobjectsfile]
... (后面还有很多,不一一列举)
四、为什么需要引入javax.*包
下面的代码是另一篇文章equinox实现Class Loader机制的代码解读提过的,bundle类载入的流程图片段代码。
...
StringpkgName=getPackageName(name);
//followtheOSGidelegationmodel
if(checkParent&&parent!=null)...{
if(name.startsWith(JAVA_PACKAGE))
//1)ifstartsWith"java."delegatetoparentandterminatesearch
//wewanttothrowClassNotFoundExceptionsifajava.*classcannotbeloadedfromtheparent.
returnparent.loadClass(name);
elseif(isBootDelegationPackage(pkgName))
//2)ifpartofthebootdelegationlistthendelegatetoparentandcontinueoffailure
try...{
returnparent.loadClass(name);
}catch(ClassNotFoundExceptioncnfe)...{
//wewanttocontinue
}
}
...
从上面的代码可以看出,如果不是第一种情况(包名不是以java.开始的)就使用第二个判断条件。
isBootDelegationPackage(pkgName)返回的是true(这个配置在以后再说),所以执行的是
parent.loadClass(name);
这行代码和第一个条件的代码是一样的,即还是使用parentClassLoader去加载javax.*的包。由于ClassLoader只加载了java.*的包,所以这里是不能得到javax.*包中的类的,会导致 ClassNotFoundException的错误。从上面的代码可以知道,代码中对ClassNotFoundException没有处理,而是继续执行了。分别从import package、required bundles、bundle内部classpath和dynamic import 获取javax.*的包。
通常情况下,这些javax.*的包作为required bundle提供,而且只需要提供一次,并避免在其他的bundle中再次/重复提供(export)。
PackageSource source = findImportedSource(pkgName);
if(source!=null){
//3)foundimportsourceterminatesearchatthesource
result=source.loadClass(name);
if(result!=null)
returnresult;
thrownewClassNotFoundException(name);
}
//4)searchtherequiredbundles
source=findRequiredSource(pkgName);
if(source!=null)
//4)attempttoloadfromsourcebutcontinueonfailure
result=source.loadClass(name);
//5)searchthelocalbundle
if(result==null)
result=findLocalClass(name);
if(result!=null)
returnresult;
//6)attempttofindadynamicimportsource;onlydothisifarequiredsourcewasnotfound
if(source==null){
source=findDynamicSource(pkgName);
这就是需要引入javax.*包的原因。
分享到:
相关推荐
深入理解OSGi:Equinox原理、应用与最佳实践源代码,同时附带equinox-SDK-3.8源代码。
Equinox p2详细图示讲解, RCP中如何实现版本检测,更新
NULL 博文链接:https://salever.iteye.com/blog/712772
第一部分主要介绍了OSGi和Equinox,以及OSGi标准的Eclipse实现;第二部分采用非正式教程的方式教给读者如何从零开始构建真实的Toast应用,针对每一个步骤都提供了完整的在线示例代码;第三部分由原型构建转向实际的...
equinox
Equinox是OSGI R4的一个实现,Equinox这个项目是由Eclipse 开源组织实现并发布的,也是大名鼎鼎的Java开发工具 Eclipse 的底层机制的实现,如果对Eclipse有点熟悉的话,你就知道Equinox 这个东西是多么的棒了,...
我实现的《使用Equinox开发OSGI应用程序》全部功能的代码和分发包,工程师基于Eclipse3.3的
equinox-SDK-3.7.zip equinox-SDK-3.7.zip equinox-SDK-3.7.zip equinox-SDK-3.7.zip equinox-SDK-3.7.zip equinox-SDK-3.7.zip
osgi的规范实现equinox 基于eclipse版本Oxygen的实现库,官网下载实在太慢,小水管下好后,方便大家下载!
全面解读OSGi规范,深刻揭示OSGi原理,详细讲解OSGi服务,系统地介绍Equinox框架的用法,并通过源代码分析其工作机制,包含大量可操作性极强的解决方案和最佳实践。
本文将介绍 Equinox 的 ServletBridge 项目,提供一个示例来说明如何使用 ServletBridge,并将简要分析 它的实现方法。 读者将首先了解到如何在 Servlet Container 中嵌入 OSGI,并从文章提供的例子中了了解其工作...
eclipse de osgi框架 equinox-SDK-3.6.1.part1
org.eclipse.equinox.p2.examples.rcp.cloud.rarorg.eclipse.equinox.p2.examples.rcp.cloud.rarorg.eclipse.equinox.p2.examples.rcp.cloud.rar
Equinox
org.eclipse.equinox.p2.examples.rcp.prestartupdate.rarorg.eclipse.equinox.p2.examples.rcp.prestartupdate.rarorg.eclipse.equinox.p2.examples.rcp.prestartupdate.rar org.eclipse.equinox.p2.examples.rcp....
OSGi and Equinox, Creating Highly Modular Java Systems
OSGI标准实现框架equinox的SDK,版本是3.8.2.里面包含equinox的所有插件jar包,适合做OSGI的开发。
equinox-SDK-3.6.1.part2
equinox-SDK-4.18.zip