java中的classloader的原理

最近在琢磨btrace,顺便也看了下JDK的classloader的原理

先直接把代码贴出来:
首先是main方法:
package com.gao;

import com.wang.MyClass;

/**
* User: wangchen.gpx
* Date: 12-12-28
* Time: 上午9:35
*/
public class ClassLoaderTeste {
public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException {
MyClassLoader myclassloader = new MyClassLoader("myclassloader");
Class<?> myClass = myclassloader.loadClass("com.wang.MyClass");
Object o = myClass.newInstance();
MyClass myClass1 = (MyClass) o;
myClass1.say();
}
}

然后是自定义的classloader
package com.gao;

import java.io.*;

/**
* User: wangchen.gpx
* Date: 12-12-28
* Time: 上午9:26
*/
public class MyClassLoader extends ClassLoader {
private String loaderName;
private String path = "G:\\idea_workspace\\asmTest\\out\\production\\asmTest\\java\\java\\";

public MyClassLoader(String loaderName) {
this.loaderName = loaderName;
}

public MyClassLoader(ClassLoader parent, String loaderName) {
super(parent);
this.loaderName = loaderName;
}

@Override
public Class<?> loadClass(String name) throws ClassNotFoundException {
return this.findClass(name);
}

@Override
protected Class<?> findClass(String name) throws ClassNotFoundException {
System.out.println("find class");
byte[] bytes = this.loadClassData(name);
return this.defineClass(name, bytes, 0, bytes.length);
}

private byte[] loadClassData(String name) {
try {
name = name.replace(".", "//");
FileInputStream fileInputStream = new FileInputStream(new File(path + name + ".class"));
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
int b = 0;
while ((b = fileInputStream.read()) != -1) {
byteArrayOutputStream.write(b);
}
return byteArrayOutputStream.toByteArray();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return null;
}
}

最后是用来装载的class:

package com.wang;

/**
* User: wangchen.gpx
* Date: 12-12-28
* Time: 上午9:08
*/
public class MyClass {
public MyClass() {
System.out.println("constructor");
}

public void say(){
System.out.println("one");
}
}


这里首先跑的话,我认为应该是直接走自己的classloader的findclass方法,但是我错了,根本就没有走到这里,所以就比较纳闷,自己看了下classloader的源码,原来是这样:

protected synchronized Class<?> loadClass(String name, boolean resolve)
throws ClassNotFoundException
{
// First, check if the class has already been loaded
//这里首先查看该类是否已经装载了,像我们自定义的类,现在还没有装载,因此这里应该返回的是null
Class c = findLoadedClass(name);
if (c == null) {
try {
if (parent != null) {
//注意这里,他首先会用父classloader去装载,父类classloader也会调用它的
//父类classloader进行装载,已经递归下去,直到进入bootstrapclassloader
//如果bootstrapclassloader找不到该类,那么他将委托为appClassloader去装载
//这个就是所说的双亲委派模型
c = parent.loadClass(name, false);
} else {
//这一步就是最后到达的bootstrapclassloader
c = findBootstrapClassOrNull(name);
}
} catch (ClassNotFoundException e) {
// ClassNotFoundException thrown if class not found
// from the non-null parent class loader
}
//注意这里,只有真的找不到的时候才会进入这一步,调用自己实现的findclass方法来loadclass
if (c == null) {
// If still not found, then invoke findClass in order
// to find the class.
c = findClass(name);
}
}
if (resolve) {
resolveClass(c);
}
return c;
}


所以这次又对类加载的机制有了一定的了解,主要是双亲委派模型的实际含义
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值