王洪伟的专栏

http://blog.teamlet.org

用户操作
[即时聊天] [发私信] [加为好友]
王洪伟ID:teamlet
169107次访问,排名451好友1人,关注者59
10年软件开发设计经验,专注J2EE领域的技术架构和应用.
teamlet的文章
原创 100 篇
翻译 9 篇
转载 67 篇
评论 146 篇
teamlet的公告

本站采用创作共用版权协议, 要求署名、非商业用途和相同方式共享. 转载本站内容必须也遵循“署名-非商业用途-相同方式共享”的创作共用协议.

关注SOA技术的发展,跟进SCA技术的理论和实现,努力实践。愿与同行者一起分享,互相勉励,共同进步。
最近评论
myself:<configuration>
<source>1.5</source>
<target>1.5</target>
<maxmem>256M</maxmem>
<encoding>UTF-8</encoding>
<……
zhi:您好!!
能不能给我也发一份源码过来!谢谢
zhi@tuanke.net
lai:2008-9-4 13:52:56 org.apache.catalina.core.ApplicationDispatcher invoke
严重: Servlet.service() for servlet jsp threw exception
java.lang.IllegalStateException: getOutputStream() has alread……
lai:2008-9-4 13:52:56 org.apache.catalina.core.ApplicationDispatcher invoke
严重: Servlet.service() for servlet jsp threw exception
java.lang.IllegalStateException: getOutputStream() has alread……
lai:2008-9-4 13:52:55 org.apache.catalina.core.StandardWrapperValve invoke
严重: Servlet.service() for servlet AxisServlet threw exception
java.lang.NullPointerException
at org.apache.axi……
文章分类
收藏
    相册
    资源联接
    Apache Tuscany
    Cruise Control
    Open CSA
    OSOA
    SOA Tools Project
    Theserverside
    中国Java开发网
    满江红
    知识共享@中国大陆
    左邻右舍
    donews的blog
    msn的blog
    Tuscany中文社区
    我用Subversion
    存档
    软件项目交易
    订阅我的博客
    XML聚合  FeedSky
    订阅到鲜果
    订阅到Google
    订阅到抓虾
    订阅到BlogLines
    订阅到Yahoo
    订阅到GouGou
    订阅到飞鸽
    订阅到Rojo
    订阅到newsgator
    订阅到netvibes

    原创 equinox实现Class Loader机制的代码解读(2)收藏

    新一篇: equinox实现Class Loader机制的代码解读(3) | 旧一篇: equinox实现Class Loader机制的代码解读(1)

     

     

     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,它是这样定义的

    private static class ParentClassLoader extends ClassLoader {
     
    protected ParentClassLoader() {
      
    super(null);
     }
    }

    这是一个空的classloader,它直接继承了java.lang.ClassLoader 。java包中的所有类都是由java.lang.ClassLoader载入的,根据类的委派模型可以知道,继承了ClassLoader就可以直接获得需要的java类。

                if (name.startsWith(JAVA_PACKAGE))
                  
    // 1) if startsWith "java." delegate to parent and terminate search
                  
    // we want to throw ClassNotFoundExceptions if a java.* class cannot be loaded from the parent.
                    return parent.loadClass(name);

    上面代码是bundle加载class的流程图中的第一步,通过parentClassLoader获得所需java.*包中的类,如下图所示。

     这就是bundle中不需要引入java.*包的原因。

    三、如何知道java.*这些类已经被装载了呢?

    1、写一个类如下:

    public class  A
    {
        
    public static void main(String[] args) 
        {
            System.out.println(
    "Hello World!");
        }
    }

    2、编译

    javac A.java

    3、运行

    java -verbose A

    显示的部分内容如下:

    [Loaded java.lang.Object from shared objects file]
    [Loaded java.io.Serializable from shared objects file]
    [Loaded java.lang.Comparable from shared objects file]
    [Loaded java.lang.CharSequence from shared objects file]
    [Loaded java.lang.String from shared objects file]
    [Loaded java.lang.reflect.GenericDeclaration from shared objects file]
    [Loaded java.lang.reflect.Type from shared objects file]
    [Loaded java.lang.reflect.AnnotatedElement from shared objects file]
    [Loaded java.lang.Class from shared objects file]
    [Loaded java.lang.Cloneable from shared objects file]
    [Loaded java.lang.ClassLoader from shared objects file]
    ... (后面还有很多,不一一列举)

     

    四、为什么需要引入javax.*包

    下面的代码是另一篇文章 equinox实现Class Loader机制的代码解读  提过的,bundle类载入的流程图片段代码。

      ...
      String pkgName 
    = getPackageName(name);
            
    // follow the OSGi delegation model
            if (checkParent && parent != null) ...{
                
    if (name.startsWith(JAVA_PACKAGE))
                  
    // 1) if startsWith "java." delegate to parent and terminate search
                  
    // we want to throw ClassNotFoundExceptions if a java.* class cannot be loaded from the parent.
                    return parent.loadClass(name);
                
    else if (isBootDelegationPackage(pkgName))
                    
    // 2) if part of the bootdelegation list then delegate to parent and continue of failure
                    try ...{
                        
    return parent.loadClass(name);
                    } 
    catch (ClassNotFoundException cnfe) ...{
                        
    // we want to continue
                    }
            }
     ...

    从上面的代码可以看出,如果不是第一种情况(包名不是以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) found import source terminate search at the source
                result = source.loadClass(name);
                
    if (result != null)
                    
    return result;
                
    throw new ClassNotFoundException(name);
            }
            
    // 4) search the required bundles
            source = findRequiredSource(pkgName);
            
    if (source != null)
                
    // 4) attempt to load from source but continue on failure
                result = source.loadClass(name);
            
    // 5) search the local bundle
            if (result == null)
                result 
    = findLocalClass(name);
            
    if (result != null)
                
    return result;
            
    // 6) attempt to find a dynamic import source; only do this if a required source was not found
            if (source == null) {
                source 
    = findDynamicSource(pkgName);
    这就是需要引入javax.*包的原因。

    发表于 @ 2008年03月16日 21:47:00|评论(loading...)|编辑

    新一篇: equinox实现Class Loader机制的代码解读(3) | 旧一篇: equinox实现Class Loader机制的代码解读(1)

    评论:没有评论。

    发表评论  


    登录
    Csdn Blog version 3.1a
    Copyright © teamlet