java类如何加载
java加载二进制流
通过一个类的全限定名来获取此类的二进制字节流
将字节流所代表的静态存储结构转化为方法区的运行时数据结构
在内存中生成一个对应这个类的java.lang.Class对象,作为方法区这个类的各种数据访问的入口。
类加载器
启动类加载器(由C++实现,是虚拟机的一部分)
其他类加载器(由java语言实现,独立于虚拟机外,都继承于java.lang.ClassLoader)
双亲委派
protected Class<?> loadClass(String name, boolean resolve)
throws ClassNotFoundException
{
synchronized (getClassLoadingLock(name)) {
// First, check if the class has already been loaded
Class<?> c = findLoadedClass(name);
if (c == null) {
long t0 = System.nanoTime();
try {
if (parent != null) {
c = parent.loadClass(name, false);
} else {
c = findBootstrapClassOrNull(name);
}
} catch (ClassNotFoundException e) {
// ClassNotFoundException thrown if class not found
// from the non-null parent class loader
}
if (c == null) {
// If still not found, then invoke findClass in order
// to find the class.
long t1 = System.nanoTime();
c = findClass(name);
// this is the defining class loader; record the stats
sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
sun.misc.PerfCounter.getFindClasses().increment();
}
}
if (resolve) {
resolveClass(c);//连接
}
return c;
}
}
一般情况下当类加载器加载一个类的时候,会先检查当前类是否已经被加载过了,如果这个类没有被加载过,则会检查当前对象是否持有父加载器,存在父加载器的时候检查父加载器是否加载过此类,如果不存在父加载器的时候则直接使用本地方法findBootstrapClass。如果没有找到这个类,就使用当前加载器加载此类
自定义一个java.lang.String是否可以被加载起来
java.*的类是由启动类加载器
通过双亲委派可知自定义的java.lang.String是不会被加载起来的,使用这个类的的时候依然是jdk的java.lang.String类