类加载器就是用来将class文件加载到内存中的一个java类!

系统默认有三个类加载器!


BootStrap:这不是java类,是java虚拟机在启动的时候加载的第一个类,这个加载器用来加载核心类,jdk安装目录\jre\lib\rt.jar下的类由这个类加载器加载!


ExtClassLoader:是java类,加载jdk安装目录\jre\lib\ext目录下的jar文件;


AppClassLoader: 加载CLASSPATH目录下的jar文件;


===================


set path:查看path环境变量的值;set path=d:\;%path%,给环境变量新增一个路径,并且保留原来的路径;path值是windows搜索可执行文件路径;


set classpath:查看classpath的值,在dos窗口设置的值是临时的;set classpath=d:\ ;路径有分号结尾时,虚拟机先去classpath路径下搜索需要的class文件,如果找不到,则在当前目录下找;当classpath路径没有分号结尾时,表示虚拟机只去指定的目录下找,找不到就报异常!如果需要在当前目录下诏,则可以这样设置 set classpath=.;d:\  (没有分号结尾!)


=========================


下面演示各个类加载器:

public class ClassLoaderTest {
     
     public static void main(String[] args) {
          ClassLoader loader = ClassLoaderTest.class .getClassLoader();
           while(loader!=null ){
              System. out.println(loader);
              loader=loader.getParent(); //得到父加载器;
          }
           /*输出结果
          sun.misc.Launcher$AppClassLoader@61e63e3d
          sun.misc.Launcher$ExtClassLoader@53004901*/
          
           /**  
           * 当把该类export ->java-> jar files->选择路径:D:\Develop\Java\jdk1.8.0_05\ jre\lib \ext\oterman.jar
           * 即将该类放于ExtClassLoader加载的目录时,此时的输出为:
           * sun.misc.Launcher$ExtClassLoader@15db9742
           * 注意需要选择run configurations 的JRE为上述导出的 jre,否则不会达到我们指定的效果;不能选择 myeclipse自带的jre!
           * 类加载器的机制是先找父类去帮忙加载,父类在自己的目录加载失败后才交给子类!
           */
     }
}




类加载器的委托机制:



当Java虚拟机要加载一个类时,到底派出哪个类加载器去加载呢?
首先当前线程的类加载器去加载线程中的第一个类。
如果类A中引用了类B,Java虚拟机将使用加载类A的类装载器来加载类B。
还可以直接调用ClassLoader.loadClass()方法来指定某个类加载器去加载某个类。
每个类加载器加载类时,又先委托给其上级类加载器。
当所有祖宗类加载器没有加载到类,回到发起者类加载器,还加载不了,则抛ClassNotFoundException,不是再去找发起者类加载器的儿子,因为没有getChild方法,即使有,那有多个儿子,找哪一个呢?
对着类加载器的层次结构图和委托加载原理,解释先前将ClassLoaderTest输出成jre/lib/ext目录下的itcast.jar包中后,运行结果为ExtClassLoader的原因。

========================



ClassLoader:



抽象类有



loadClass():该方法实现了委托父类加载类的机制,如果如果找不到,会调用findClass();



findClass():自己实现该方法返回class,可以调用defineClass()按照自己的需求去返回一个类的字节码!



defineClass(String name, byte[] b, int off, int len):将一个 byte 数组转换为 Class 类的实例。

调用时则:


MyClassLoader.loadClass()返回class对象


自定义ClassLoader:需要将这个classloader挂载搞加载树上去,即需要指定父加载器!重载有参数的构造方法即可!

======================== 
   
 
   

     ClassLoader:


抽象类有


loadClass():该方法实现了委托父类加载类的机制,如果如果找不到,会调用findClass();


findClass():自己实现该方法返回class,可以调用defineClass()按照自己的需求去返回一个类的字节码!

defineClass(String name, byte[] b, int off, int len):将一个 byte 数组转换为 Class 类的实例。


调用时则:



MyClassLoader.loadClass()返回class对象


自定义ClassLoader:需要将这个classloader挂载搞加载树上去,即需要指定父加载器!重载有参数的构造方法即可!



例:下面这个例子将byte[]加载为class文件!

/**
 
    
 * 自定义类加载器,将byte[]加载为class对象
 
    
 * 
       
     @author 
       
     oterman
 
    
 *
 
    
 */
 
    
class 
       
     MyClassLoader 
       
     extends 
       
     ClassLoader{
 
    
      
     //自定父加载器,按照加载机制进行加载!加载一个类时,都先交给父类加载,父类加载不了,才一级一级往下传!
 
    
      
     public 
       
     MyClassLoader(ClassLoader parent) {
 
    
           
       
     super 
     (parent);
 
    
     }
 
    

 
    
      
     public 
       
     Class getClass( 
     byte 
     [] bytes){
 
    
          System. 
       
     out 
     .println( 
     "myclassloader2...." 
       
     );
 
    
           
       
     return 
       
     defineClass( 
     null 
     , bytes, 0, bytes. 
     length 
     );
 
    
     }
 
    
}
 
   

     ================= 
   
 
   

     使用自定义的classloader: 
   
 
   
      
     //将硬盘上的class文件读取为byte[];
 
    
     FileInputStream fis = 
       
     new 
       
     FileInputStream(dir);
 
    
     ByteArrayOutputStream bos = 
       
     new 
       
     ByteArrayOutputStream();
 
    
      
     byte 
     [] buffer = 
       
     new 
       
     byte 
       
     [1024];
 
    
      
     int 
       
     len = 0;
 
    
      
     while 
       
     ((len = fis.read(buffer)) != -1) {
 
    
          bos.write(buffer, 0, len);
 
    
     }
 
    
     bos.close();
 
    
      
     byte 
     [] bytes = bos.toByteArray(); 
       
     //class文件的字节码
 
    
     
 
    
      
     //自定义classloader,将byte[]翻译成class对象
 
    
      MyClassLoader classloader = 
       
     new 
       
     MyClassLoader( 
     this 
     .getClass().getClassLoader());
 
    
     Class clazz =classloader.getClass(bytes);
 
    
     
 
    
      
     //判断该类上是否有注解,如果有,则放入map中去;
 
    
      
     Control 
       
     annotation = ( 
       
     Control 
     ) clazz.getAnnotation( 
     Control 
     . 
     class 
     ) ;
 
    
      
     if 
       
     (annotation != 
       
     null 
     ) {
 
    
          String uri = annotation.value();
 
    
           
       
     actionMap 
     .put(uri, clazz.newInstance());
 
    
     }