探究1:怎样“new”一个对象
例子:通过 new 创建 Car 实例
class文件被ClassLoader加载到JVM虚拟机同时初始化生成Car Class文件,
该文件为Car实例的模板,所有实例创建都是依据该模板创建的。
所以在我们使用“new”关键字创建对象的时候,其实得到的都是“Car Class” 的副本。
探究2:双亲委派机制
四种加载器
idea下查看源码了解执行流程
ctrl+N,搜索ClassLoader,然后选java.lang下的类
进入该类后ctrl+F,搜索loadClass
protected Class<?> loadClass(String name, boolean resolve)
throws ClassNotFoundException
{
synchronized (getClassLoadingLock(name)) {
// 首先检查这个classsh是否已经加载过了
Class<?> c = findLoadedClass(name);
if (c == null) {
long t0 = System.nanoTime();
try {
// c==null表示没有加载,如果有父类的加载器则让父类加载器加载
if (parent != null) {
c = parent.loadClass(name, false);
} else {
//如果父类的加载器为空 则说明递归到bootStrapClassloader了
//bootStrapClassloader比较特殊无法通过get获取
c = findBootstrapClassOrNull(name);
}
} catch (ClassNotFoundException e) {}
if (c == null) {
//如果bootstrapClassLoader 仍然没有加载过,则递归回来,尝试自己去加载class
long t1 = System.nanoTime();
c = findClass(name);
sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
sun.misc.PerfCounter.getFindClasses().increment();
}
}
if (resolve) {
resolveClass(c);
}
return c;
}
}
类加载流程
该图先上后下,先左后有执行
双亲委派机制
1.防止重复加载同一个.class
。通过委托去向上面问一问,加载过了,就不用再加载一遍。保证数据安全。
2.保证核心.class
不能被篡改。通过委托方式,不会去篡改核心.clas
,即使篡改也不会去加载,即使加载也不会是同一个.class
对象了。不同的加载器加载同一个.class
也不是同一个Class
对象。这样保证了Class
执行安全。
沙箱隔离是目标,双亲委派是实现。
从这幅图不难看出,java、javax、sun等包下的类(如java.lang.Object、java.lang.String类等)都是被启动类加载器所加载的,所以我们在是用这些类的时候并不需要导包,并且我们自己创建一样路径下创建同名的类,是加载不到的。
++++++++++++++++++++++++++++++++++
+ 如有问题可+Q:1602701980 共同探讨 +
++++++++++++++++++++++++++++++++++