java基础学习笔记之类加载器
类加载器
java类加载器就是在运行时在JVM中动态地加载所需的类,java类加载器基于三个机制:委托,可见,单一。
把classpath下的那些.class文件加载进内存,处理后成为字节码,这些工作是类加载器做的。
- 委托机制指的是将加载类的请求传递给父加载器,如果父加载器找不到或者不能加载这个类,那么再加载他。
- 可见性机制指的是父加载器加载的类都能被子加载器看见,但是子加载器加载的类父加载器是看不见的。
- 单一性机制指的是一个类只能被同一种加载器加载一次。
默认类加载器
系统默认三个类加载器:
- BootStrap
- ExtClassLoader
- AppClassLoader
类加载器也是java类,而BootStrap不是。验证代码:
publicclassClassLoaderTest{ publicstaticvoidmain(String[]args){ System.out.println(System.class.getClassLoader()); } }
输出:null
如果使用System.out.println(System.class.getClassLoader().toString);,则报空指针异常:
Exceptioninthread"main"java.lang.NullPointerException atcom.iot.classloader.ClassLoaderTest.main(ClassLoaderTest.java:10) atsun.reflect.NativeMethodAccessorImpl.invoke0(NativeMethod) atsun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) atsun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) atjava.lang.reflect.Method.invoke(Method.java:483) atcom.intellij.rt.execution.application.AppMain.main(AppMain.java:144)
可见,System类是由BootStrap类加载器加载。
类加载器的委托机制
类加载器的树状图
类加载器
一般加载类的顺序:
- 首先当前线程的类加载器去加载线程中的第一个类
- 如果类A应用了类B,java虚拟机将使用加载类A的类加载器来加载类B
- 还可以直接调用ClassLoader.loadClass()方法来制定某个类加载器去加载某个类
自定义类加载器的编写原理
API:
ClassClassLoader
模板方法设计模式
父类:
loadClass(类加载的流程,模板)
findClass供子类覆盖的、被loadClass方法调用的类加载逻辑
defineClass得到class文件转换成字节码
子类:覆盖findClass方法
例子:
loadClass方法的源码
protectedClass<?>loadClass(Stringname,booleanresolve) throwsClassNotFoundException { synchronized(getClassLoadingLock(name)){ //First,checkiftheclasshasalreadybeenloaded Class<?>c=findLoadedClass(name); if(c==null){ longt0=System.nanoTime(); try{ if(parent!=null){ c=parent.loadClass(name,false); }else{ c=findBootstrapClassOrNull(name); } }catch(ClassNotFoundExceptione){ //ClassNotFoundExceptionthrownifclassnotfound //fromthenon-nullparentclassloader } if(c==null){ //Ifstillnotfound,theninvokefindClassinorder //tofindtheclass. longt1=System.nanoTime(); c=findClass(name); //thisisthedefiningclassloader;recordthestats sun.misc.PerfCounter.getParentDelegationTime().addTime(t1-t0); sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1); sun.misc.PerfCounter.getFindClasses().increment(); } } if(resolve){ resolveClass(c); } returnc; } }
API文档中的例子:
classNetworkClassLoaderextendsClassLoader{ Stringhost; intport; publicClassfindClass(Stringname){ byte[]b=loadClassData(name); returndefineClass(name,b,0,b.length); } privatebyte[]loadClassData(Stringname){ //loadtheclassdatafromtheconnection ... } }