类加载器就是用来将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的原因。
首先当前线程的类加载器去加载线程中的第一个类。
如果类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挂载搞加载树上去,即需要指定父加载器!重载有参数的构造方法即可!
例:下面这个例子将byte[]加载为class文件!
========================
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());
}