类加载器重要方法
实现自定义类加载器,需要继承ClassLoader
并重写findClass
方法并会用到ClassLoader
的defineClass
和loadClass
。
findClass(String name)
根据指定的二进制文件名称来查找类。
这个方法应该被类加载器实现覆盖,该实现遵循委托模型来加载类,并且在检查所请求的类的父类加载器后,将由loadClass方法调用。默认实现会抛出一个ClassNotFoundException。
defineClass(String name, byte[] b, int off, int len)
将字节数组转换为类的实例,在使用Class之前,必须先解析它。
该方法为新定义的类分配一个默认的ProtectionDomain
,ProtectionDomain
实际上被授予返回的同一组权限当Policy.getPolicy().getPermissions(new CodeSource(null, null))
被调用。(确保返回的Class一切都是正确)。默认域在defineClass
的第一次调用时创建,并在随后的调用中重新使用。
要为类指定特定的ProtectionDomain
,请使用defineClass(String,byte [],int,int,java.security.ProtectionDomain)defineClass
方法将一个 ProtectionDomain
作为其参数之一。
@param name:期待二进制的名字,如果不知道也可以传入一个null
.
@param b:构成类数据的字节,介于 off 到off + len-1 的字节应该具有由定义的有效类文件的格式,Java虚拟机规范。
@param off:类数据的 b 中的起始偏移量
@param len:类数据的长度。
@return:从指定的类数据创建的Class对象。
@throws ClassFormatError:如果数据不包含有效的类
@throws IndexOutOfBoundsException:如果off或len为负数,或者off + len大于b.length 。
@throws SecurityException:如果尝试将此类添加到某个包里,被添加的包,包含了一些类,这些类是由不同签名的类构成的与此类的签名不同,或者是我们尝试在一个包中定义一个类,然后包的名以java.
开头的,都会抛出此异常。
loadClass(String name)
加载拥有指定二进制文件名字的类。
此方法搜索类与loadClass(String, boolean)
方法具有相同的方式。它由Java虚拟机器调用来解析类的引用。调用此方法等效于调用loadClass(String, boolean)
方法。
loadClass(String name, boolean resolve)
加载拥有指定二进制文件名字的类。此方法的默认实现会按照以下顺序搜索类:
1.调用findLoadedClass(String)
来检查类*是否已被加载。
2.调用父类加载器上loadClass
方法,如果父亲是null
那么内建的虚拟机就会启动调用。
3.调用findClass(String)
方法来查找类。
如果使用上述步骤找到该类,并且resolve
标志为true
,这个方法将会对结果类对象调用resolveClass(Class)
方法。
鼓励ClassLoader
的子类重写findClass(String)
,而不是用此方法。
除非重写,否则此方法会在整个类加载过程中同步getClassLoadingLock
方法的结果.
网友评论