写程序的时候,时常会思考,我们的类是哪来的?为何我们想用sdk中的类,就有这个类呢?答案肯定是classloader读进虚拟机来的。那他究竟是怎么被加载到虚拟机中的呢?看看如何定制classloader的代码,就相当一目了然了。
main() {
runClass(name, key);
}
public void runClass(String name, int key){
ClassLoader classLoader = new CryptoClassLoader(key);
Class<?> c = classLoader.loadClass();
Method m = c.getMethod("main", String[].class);
m.invoke(null, (Object)new String[]{});
}
class CryptoClassLoader extends ClassLoader{
int key;
CryptoClassLoader(int key){
this.key = key
}
protected Class<?> findClass(String name) throws ClassNotFoundException{
byte[] classBytes = null;
try{
classBytes = loadClassBytes(name);
} catch(IOException e) {
throw new ClassNotFoundException();
}
Class<?> cl = defineClass(name, classBytes, 0, classBytes.length);
if (cl == null) throw new ClassNotFoundException(name);
return cl;
}
private byte[] loadClassBytes(String name) throws IOException{
String cname = name.replace(".","/") + ".caesar"; //(我怎么觉得应该是.class? 不太懂这里...不会印刷错误吧?)
FileInputStream in = null;
in = new FileInputStream(cname);
try{
ByteArrayOutputStream buffer = new ByteArrayOutputStream();
int ch;
while((ch = in.read()) != -1){
byte b = (byte)(ch - key);
buffer.write(b);
}
return b.toByteArray();
} finally {
in.close();
}
}
}
此段代码看完发现(defineClass方法),虚拟机加载新类的方式就是:使用父类的
defineClass方法将某Class类的字节码传递给虚拟机,(其实就是以字节形式把字节码传递给虚拟机),并且创建相应的类。
下来解释一下上面的程序:key这个int。这个例子使用了一个小的加密方法,就是类文件中所有的字节都向后移动了key个位置,比如3,那么,在加载此类时,就应该向前移动3,来保证文件被解密。这里我们就引申出了自定义classloader的
用法之一,即
加密。你想使用这个类?请知道密钥先!就好像游戏的注册码一样,没有key和我们的classloader,你拿到类文件也是没有任何意义的。(好吧我承认程序里的加密方法不怎么先进)
怎么样,很棒吧。其实一直以来,还有个疑问,如果包名和类名一样的情况,classloader应该如何处理?首先,像上面的情况,一个路径下是不能放相同名字的类文件的,先排除此情况。那如果在多jar包的环境下,真的有两个一样的类名呢?我想、估计是直接使用第一个被load进的类吧,因为classloader会判断此类已经被加载进来了,毕竟classloader也不会读心数,知道你想使用哪个jar包的类。
其实,如果类名和包名都重复,我们可以看见,如果classloader不一样,你那么也不会有任何问题,它们之间是不会相互干扰的,比如Web中的每个applet,就是由不同的classloader加载的,所以,命名冲突在applet之间是允许的。就好像不同的
命名空间。
还有个比喻,我想到的,可能会比较烂。比如,android上的两个不同的apk,如果两个程序中,使用相同的包名和类名的话,应该也不会相互干扰,因为不同的classloader导致。(因为每个进程都有自己的虚拟机,那么classloader也理应不同,恩,应该是这样,本身classloader就是给虚拟机服务的嘛。)
分享到:
相关推荐
Java 虚拟机中ClassLoader 相关简介 双亲委托机制 Android 中ClassLoader 简介
ClassLoader运行机制 自己写的ClassLoader运行机制 自己写的ClassLoader运行机制 自己写的ClassLoader运行机制 自己写的ClassLoader运行机制 自己写的
自定义classloader的使用
ClassLoader原理,ClassLoader原理 ClassLoader原理
主要内容包括 Java类加载机制及加载流程,以及如何定义自己的类加载器,如何实现类的热替换。
Java ClassLoader定制实例
破解java加密的ClassLoader.java,在classloader植入破解代码
Classloader
java classloader classpath 张孝祥
理解Java ClassLoader机制
用于验证理解Android中Classloader加载类机制的程序demo,从中可以对比DexClassLoader和PathClassLoader的区别联系。
ClassLoader类加载机制和原理详解
classloader 源码,自定义classloader
自定义ClassLoader,控制台输入调试。 运行期间 重新载入指定目录的class文件。可实现对于类的功能函数更新。 用到java 反射,@interface 等技术
关于J2EE服务器的ClassLoader的原理,该文档清晰了揭示了jvm装载类的顺序,同时用户可以自定义修改classLoader的配置 通过该文档,可以加深对Java虚拟机的理解
classloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloaderclassloader
JVM内存模型,类加载模式工作机制详细,内存屏障,类从被加载到虚拟机内存中开始,直到卸载出内存为止,它的整个生命周期包括了:加载、验证、准备、解析、初始化、使用和卸载这7个阶段。其中,验证、准备和解析这三...
JVM ClassLoader简析.压缩包中文档和示例代码