核心提示:如果你想对java加载类的机制有比较深入的理解的话,请你花点时间好好阅读一下这篇东西. 这里从三个方面讲述java classloader 1,翻译一下classloader的相关技术文章(来自Core Java第二卷的Chapter 9. Security),由于我的英文水平有限,难免有不合适,请大家指出.
如果你想对java加载类的机制有比较深入的理解的话,请你花点时间好好阅读一下这篇东西.
这里从三个方面讲述java classloader
1,翻译一下classloader的相关技术文章(来自Core Java第二卷的Chapter 9. Security),由于我的英文水平有限,难免有不合适,请大家指出.
2,将从web应用出发模拟一下web应用的classloader(这节是重点)
3,通过测试2实现的classloader来加深理解
以下是翻译的关于ClassLoader介绍
ClassLoader
JAVA编译器把源代码转换成一个假想机器(就是我们所说的虚拟机)的语言.虚拟机指令被保存在class后缀的文件里.
每一个类文件包含类和接口的定义以及实现代码.这些类文件必须被一个程序解释,这个程序能够把虚拟机的指令翻译成
宿主机的机器语言.
注意:虚拟机只加载执行一个程序所需要的类文件.举个例子,比如执行MyProgram.class,虚拟机运行的步骤如下:
1,虚拟机有一个加载类文件的机制,比如,从硬盘读取文件或者就网络获得;虚拟机用这个机制加载MyProgram的类文件
2,如果MyProgram有一个实例变量或者是超类,那么实例变量和超类的类文件也被加载.
(加载一个类所依赖的所有类的过程叫做resolving the class-->自己理解吧)
3,然后虚拟机执行MyProgram的main方法(因为是静态方法,所以不需要new MyProgram的实例)
4,如果main 方法或者main方法调用的方法需要其他的类的话,这些类也被加载.
类加载机制不是仅仅用一个类加载器,任何一个java程序至少有以下三个类加载器(为了不影响大家的理解,这里我就不翻译这三个类加载器的名称了)
The bootstrap class loader:加载系统类(有代表性的,jdk的rt.jar里的类),他是虚拟机的必要组成部分,并且一般是用C实现的.
也有类加载器对象(就是指具体的一个类加载器)不关联bootstrap class loader,比如String.class.getClassLoader()返回null.
The extension class loader:加载jre/lib/ext目录下的class,你可以把你的jar文件放到这个目录,extension class loader
将会加载到jar里面的类,即使你不设置classpath.(一些人建议使用这个机制以让你不受classpath的烦扰,不过注意以下的警告)
The system class loader (有时也叫应用程序加载器):加载应用程序类.
他主要加载classpath目录和jar/zip文件里的class,通过设置CLASSPATH环境变量或者是运行java的时候用[-classpath]选项指定classpath
在SUN的java实现里,the extension and system class loaders都是用java实现的,他们都是URLClassLoader类的实例.
警告:如果你把jar文件放到jre/lib/ext目录下,并且你的jar文件中的类需要加载一个不是system or extension的类的话,
你将遇到麻烦.扩展类加载器不使用类路径.如果你想把类放到jre/lib/ext下进行管理的话,请牢记这一点.
==>怎么理解这一点:也就是说如果你把自己的x.jar放到jre/lib/ext下的话,如果你自己的x.jar里的class要用到不在
x.jar里也不在jre/lib/ext的类的话,会导致类加载不了.不难想象吧,因为你x.jar里的类是由extension class loader
加载的,他不会加载classpath路径下的类.
警告:把jar文件放到jre/lib/ext目录下,还有第二个缺陷:有时侯,程序员忘记了他很久以前放在这个目录下的类文件.
当class loader似乎忽略了类路径(其实没有,因为类加载总是先让父的类加载器加载类,只有父的类加载器加载不了的话
才由自己来加载,"extension class loader是system class loader的父,因此..."),
而加载了放在扩展目录下的遗忘已久的类的时候,他们就会迷惑不解.
class loader有父子关系,bootstrap class loader以外的每一个class loader都有一个父的类加载器.
类加载器会给父的加载器一个机会加载任何给定的类,如果父加载器加载失败的话自身才去加载.
举个例子,当系统class loader被要求加载一个系统类的时候(比如,java.util.ArrayList),
那么,首先需要extension class loader加载,而extension class loader又先让bootstrap class loader,
最终由bootstrap class loader查找并且加载了rt.jar,其他任何类加载器不需要再搜索.
注意:当实现一个类加载器的时候,你应该总是授权父加载器去加载类.否则,将会有一些潜在的安全隐患:自定义的类加载器
可能避开重要的安全检查,意外地加载了系统类.
Applets, servlets, and RMI stubs是用户自定义的类加载器加载的.你甚至可以根据自己的需要写自己的类加载器.
这种方式允许你在传字节码给虚拟机之前实现特殊的安全检查.比如,你可以写一个类加载器拒绝加载没有用"paid for"
表示的类.下一节将展示这么去实现.
大多数时间,你不需要担心类加载器.很多类因为被其他类引用而被加载,这个过程对你来说是透明的.
如果你在程序里调用Class.forName来加载一个类,那么一个新的类被调用Class.forName的类的加载器加载.通常,
这不会有什么问题.然而,在下面的情况下将会失败:
1.你的lib自己实现了一个有Class.forName方法的类的时候
2.你的lib里的类的方法被一个不同的类加载器加载的应用程序类所调用的时候(这一先需要好好理解)
3,被加载的类对于应用程序的类加载器来说是不可见的时候(也就是说classpath下没有相关的类)
这种情况下.库类需要搜索应用程序类加载器(代码如下):
Thread t = Thread.currentThread();
ClassLoader loader = t.getContextClassLoader();
Class cl = loader.loadClass(className);
Using Class Loaders as Namespaces
任何一个java程序员都知道包名是用来消除名字冲突的.在标准类库里有两个叫Date的类(java.util.Date and java.sql.Date).
简单的名字(这里指的是你在程序里直接写Date)只是程序员方便,并且需要包含import语句.在一个运行的程序中,
所有的class都包含他们的包名.
这也许让你吃惊,然而,在同一个虚拟机里面你可以有两个类名和包名都相同的类.一个类是通过他的全名和类加载器来标识的.
This technique is useful for loading code from multiple sources.比如,浏览器为每个web页面使用单独的
applet class loader.这允许虚拟机分开来自不同web页面的类,不管他们是怎么命名的.
NOTE:这项技术还有其他的用途,比如Sevlet和EJB的"热部署",请参照:http://java.sun.com/developer/TechTips/2000/tt1027.html
Writing Your Own Class Loader
The loadClass method of the ClassLoader superclass takes care of the delegation to
the parent and calls findClass only if the class hasn't already been loaded and
if the parent class loader was unable to load the class.
定义自己的类加载器只需要继承ClassLoader类并且重写findClass(String className)方法.
ClassLoader父类的loadClass方法负责授权给父的类加载器 并且只有在还没有加载并且父的类加载器不能加载的时候
才调用findClass方法.
NOTE:在早期版本的JDK中,程序员必须重写loadClass方法.现在不再建议这种做法.
实现findClass方法必须做下面两件事情,
1,从文件或者其他地方加载类的字节码
2,为了把字节码提交给虚拟机,需要调用ClassLoader类的defineClass方法,
以下是实现一个加载加密类文件的类加载器.
-
-
-
-
class CryptoClassLoader extends ClassLoader {
-
-
-
-
-
-
-
public CryptoClassLoader(int k) {
- key = k;
- }
-
-
protected Class findClass(String name) throws ClassNotFoundException {
-
byte[] classBytes = null;
-
try {
- classBytes = loadClassBytes(name);
-
} catch (IOException e) {
-
throw new ClassNotFoundException(name);
- }
-
-
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";
-
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);
- }
- in.close();
- padding-top: 0px !important; padding-right: 3px !important; padding-bottom: 0px !important; padding-left: 10px !important; font-family: Verdan
分享到:
相关推荐
在Java虚拟机(JVM)中,类加载器(ClassLoader)是至关重要的组成部分,它负责查找和加载类的字节码文件。理解ClassLoader的工作机制对于深入掌握Java应用程序的运行至关重要。这里我们将详细讨论ClassLoader的运行...
《深入理解ClassLoader工作机制》 Java虚拟机(JVM)中的ClassLoader是负责加载类到内存中的核心组件。它不仅承担着将字节码转换为可执行对象的重任,还参与了类生命周期的各个阶段,包括加载、验证、准备、解析、...
在Java中,Classloader是加载类的关键组件,它负责查找、加载和初始化字节码文件。自定义Classloader允许开发者根据特定需求定制类的加载逻辑,例如加密类文件、隔离不同版本的库或者动态加载代码。本文将深入探讨...
在Java编程语言中,ClassLoader是一个至关重要的组成部分,它负责加载类到JVM(Java虚拟机)中,使得程序能够执行。本示例"ClassLoader小例子"将深入探讨这个概念,并通过一个具体的程序来演示其工作原理。下面我们...
Java ClassLoader是Java运行时系统的关键但经常被忽视的组件,负责在运行时查找和加载类文件。通过创建自定义ClassLoader,你可以定制JVM,使类文件的引入方式完全重新定义,这提供了很多实用和有趣的可能。这篇教程...
Java ClassLoader机制是Java虚拟机(JVM)中一个至关重要的组成部分,它的主要任务是将类的.class文件加载到JVM中,使得程序能够运行。ClassLoader不仅负责类的加载,还涉及类的验证、初始化等一系列过程。理解...
在Java编程语言中,`ClassLoader`是一个至关重要的组件,它负责加载类到JVM(Java虚拟机)中。本文将深入探讨`ClassLoader`的工作原理、加密解密应用程序以及如何防止类被反编译。 首先,让我们理解`ClassLoader`的...
在Java编程语言中,ClassLoader是一个至关重要的组成部分,它负责加载类到JVM(Java虚拟机)中。理解ClassLoader的工作原理以及如何定制它,对于深入学习Java的运行机制和进行高级应用开发具有重要意义。本篇文章将...
在Java世界中,类加载器(ClassLoader)是关键的组件之一,它负责将类的字节码文件(.class)从文件系统或网络中加载到Java虚拟机(JVM)中,使得程序能够运行。本篇文章将深入探讨ClassLoader的关系网络以及如何...
《ClassLoader详解》 Java应用程序的运行离不开类的加载,而ClassLoader正是这个过程的关键角色。它负责将类的字节码加载到Java虚拟机(JVM)中并转换为可执行的Java对象。深入理解ClassLoader的工作原理对于优化...
Java ClassLoader机制是Java运行时环境中的核心组件之一,它负责加载类到JVM(Java虚拟机)中,使得程序能够执行。理解ClassLoader的工作原理对于优化应用性能、处理类加载问题以及实现自定义加载器至关重要。 首先...
在Java编程语言中,ClassLoader是核心组件之一,它负责加载类到JVM(Java虚拟机)中执行。本文将深入探讨ClassLoader的工作原理和类加载机制,帮助开发者理解这个至关重要的概念。 1. 类加载机制概述 Java的类加载...
在Java编程语言中,ClassLoader是一个至关重要的组成部分,它负责加载类到JVM(Java虚拟机)中执行。这篇测试主要探讨了ClassLoader的工作原理及其在实际应用中的使用。通过阅读给出的博文链接,我们可以深入理解...
### Java ClassLoader与ClassPath详解 #### 一、概述 在Java编程中,类加载机制是十分关键的一个环节。类加载器(`ClassLoader`)负责将编译后的`.class`文件加载到Java虚拟机(JVM)中执行,而类路径(`ClassPath...
Java的类加载器(ClassLoader)体系结构是JVM(Java虚拟机)中至关重要的一部分,它负责将类的字节码转换为运行时的类实例。本文将深入探讨启动类加载器、扩展类加载器、系统类加载器以及用户自定义类加载器,同时还...
破解java加密的ClassLoader.java,在classloader植入破解代码