(由于是边看边写,有些内容是根据已有知识推理得来,不一定正确,请各位指正,谢谢)
--是何物
说ClassLoader就得先说Class。一个Class实例就代表JAVA中的一个类、接口、原始数据类型,甚至void都有代表它的Class实例。我们可以这样来看,reference是记录了关于它对应对象的一些信息,而一个Class实例则记录了关于它代表的类或接口的信息。每一个Class实例都包含着对一个ClassLoader子类对象的应用。写到这里,现在对ClassLoader是什么有个大概的了解了。
class loader 就是用来装载类的,类的装载本是JVM做的事情,我们不用管它,但是JAVA为什么要提供ClassLoader这个类呢?由于毕业设计的缘故看了一些开源项目的源代码,最近又在看,发现代码里,ClassLoader用的挺多,于是决定要搞懂这个问题,也于是写了这篇文章。
--有何用
先回答刚才的这个问题吧,JAVA里为什么要提供ClassLoader这个类。我觉得我BLOG收藏夹里的一篇文章里说的很对,“让用户可以在只拥有程序二进制代码的情况下添加自己的功能”,凭我的经验的确是这样,通过ClassLoader我们可以文件加载为类,把某些资源加载为类,而且是在运行时实现。另外就是J2EE里也好,很多开源项目也好,越来越多的用XML文件做配置文件或者是描述符、部署符。其实这些通过XML文档描述的配置信息最终都要变成JAVA类,这些都是ClassLoader的功劳。这里说的是我们自己定义的ClassLoader的子类的用处。至于JVM对它的使用自不必说了。
--怎么用
ClassLoader使用一种委托模式查找类和资源。每一个ClassLoader的实例都会有一个父辈的ClassLoader与它相关,当有请求要查找查找类时,它就会先委托父辈的ClassLoader来查找,找不到然后才是自己查找(后面会讲到怎么查找),有时候当父辈为NULL时,JVM内置的类(称为:bootstrap class loader)就会充当父类。下面就是一种对ClassLoader中loadClass()方法的实现。
1: protected synchronized Class loadClass(String name, boolean
2: resolve)throws ClassNotFoundException{
3: // 首先, 检查这个类是否已经加载。
4: Class c = findLoadedClass(name);
5: if (c == null) {
6: try {
7: if (parent != null) {
8: c = parent.loadClass(name, false);
9: }else{
10: c = findBootstrapClass(name);
11: }
12: }catch(ClassNotFoundException e){
13: // 如果仍然没有找到,那么就调用findclass查找这个类.
14: c = findClass(name);//sarsor's note:findClass() may throw
//ClassNotFoundException why not try-catch block here
15: }
16: }
17: if (resolve) {
18: resolveClass(c);
19: }
20: return c;
}
这就是遵循的上面说到的委托模式进行的查找,所有方法都找不到class之后最后用findClass(name)寻找,所以一般我们定义ClassLoader的子类时只要覆盖findClass()就可以了,至于怎么覆盖我BLOG收藏夹里的一篇文章里有具体的实例,查找类,和查找资源的实例都有,这里就不重复说了。只是要特别注意一下里面的defineClass()方法,它能把从类文件读得的字节数组转换为Class的实例。
其实写到这里,对怎么用还是不太怎么明白,我下一步准备研究开源项目里的源代码,到时候拿实例来跟大家分享。另外,还有关于Method类的一些用法。
(读书笔记,边看边写,未完待续)