1.认识ClassLoader:
Class类描述的是整个类的信息,在Class类中提供的forName()方法,这个方法根据ClassPath配置的路径进行类的加载,若说现在你的类的加载路径可能是网络,文件,这个时候必须实现类加载器.即为ClassLoader类的主要作用.
Class类的方法:
public ClassLoader getClassLoader();
对应举例:
class Messber{}
public class Test12{
public static void main(String[] args){
Class<?> cls = Messber.class;
System.out.println(cls.getClassLoader());
System.out.println(cls.getClassLoader().getParent());
System.out.println(cls.getClassLoader().getParent().getParent());
}
}
对应结果为:
sun.misc.Launcher$AppClassLoader@18b4aac2
sun.misc.Launcher$ExtClassLoader@1540e19d
null
如上图,出现了两个类加载器,AppClassLoader(应用类加载器)与ExtClassLoader(扩展类加载器)
类加载器:JVM设计团队把类加载阶段中的"通过一个类名的全限定名来获取描述类的二进制字节流",这个动作放在java虚拟机外部实现,以便让应用程序自己决定如何去获取所需要的类.实现这个动作的代码模块称为"类加载器".
2.双亲委派模型:
举例:
Class<?> cls = findLoaded(name);
if(cls == null){
long t0 = System.nanoTime();
try{
if(parent != null){
cls = parent.LoadClass(name,false);
}else{
cls = findBootstrapClassOrNull(name);
}
}catch(ClassNotFoundException e){
}
if(cls == null){
long t1 = System.nanoTime();
cls = findClass(name);
}
}
if(resolve){
resolveClass(cls);
}
对应图为:
如上图可知:类加载器之间的关系,就被称为双亲委派模型.
工作流程:如果一个类加载器收到了类加载的请求,首先不会主动加载类,而是把请求委托给父类加载器.因此,所有的加载请求都应当传送到BootStrap加载器中.只有父类加载器无法完成加载请求,子类加载器才会尝试自己加载.
3.自定义类加载器
对应方法:
protected Class<?> loadClass(String name,boolean resolve)
默认类加载器:
class Messber{
public String toString(){
return "Member";
}
}
public class Test12{
public static void main(String[] args)throws Exception{
System.out.println(Class.forName("Member").getClassLoader().loadClass("Member").newInstance());
}
}
实现自定义加载器:
protected final Class<?> defineClass(String name,byte[] b,int off,int len);
注:用户自定义加载器,可以通过动态路径进行类的加载操作
比较两个类相等的前提:必须是同一个类加载器加载.否则即使来自同一个类,被同一虚拟机加载,加载器不同,这两个类注定不同.