常见JVM问题

https://www.jianshu.com/p/bdd7663f69c2

谈论一下你所了解的类加载器,什么是双亲委派,如何打破?

思路: 类加载器介绍 → 目的/意义 → 何为双亲委派 → 如何打破

什么是类加载器?

类加载器,根据指定的全限定类名,将.class字节码文件加载到JVM内存中,并转化为Class对象的工具。

有哪些类加载器?他们有什么关联关系?

  • 启动类加载器(Bootstrap ClassLoader) :由C++语言实现(针对Hotspot),负责将存放在<JAVA_HOME>\lib目录或者-Xbootclasspath参数指定的路径中的类库,加载到内存中。
  • 扩展类加载器(Extension ClassLoader):负责加载<JAVA_HOME>\lib\ext目录或者java.ext.dirs系统变量指定的路径中的所有类库。
  • 应用程序类加载器(Application ClassLoader):负责加载用户类路径(classpath)上的指定类库,我们可以直接使用这个类加载器。一般情况,如果我们没有自定义类加载器默认用的就是这个加载器。
  • 自定义加载器(CustomClassLoader) 用户自行定义的类加载器,可以完成如对类字节码加密的操作。
    从上到下是父类加载器和子类加载器的关系。
    ###类加载器的整个过程是怎样的?(方法调用+参数传递)

loadClass方法流程

父加载器,父类(加载器),某个加载器的加载器,他们之间的关系?

类加载器的继承关系
上图中是含有继承关系的加载器,是父类子类关系,那么是我们口中的父加载器吗?不妨看如下代码。
类加载的双亲和它的加载器的关系
从上图可以看出,对于自己定义的类,我们采用的是应用加载器进行加载。应用加载器的加载器,是null,也许是bootstrap类加载器,我们不可见。应用加载器的parent,我们说双亲,也是一个加载器,是扩展类加载器。扩展类加载器的双亲,不可见。
我们有理由总结,双亲是一个具有更执行权限的类加载器,当目前类加载器无法完成加载工作时,依托于其双气完成;和当前类并非继承关系,也非加载器的加载器关系。

双亲委派模型是怎样运行的?

一句话总结,从子到父再从父到子的过程。(与上文loadClass()方法有关)
每个类加载器都会去询问当前缓存中是否已加载类,若无则调用向父加载器的加载方法,陷入递归。当到达最顶的启动类加载器时,若缓存中也无,则会要求子加载器去完成加载的这一操作,递归返回。
双亲委派过程

为什么要有双亲委派?

两个角度考虑

  1. 安全性
    如果用户自定了一个与核心类同名的类,如java.lang.String,自己进行加载时若不向父加载器询问是否已经加载,而是直接加载,可能存在隐患。
  2. 便利性
    避免重复加载
    ###如何打破双亲委派?具体怎么做?
    继承ClassLoader类,
    重写findClass()方法
    然后再创建并加载类,传递全限定类名,创建实例,调用方法
//继承ClassLoader
public class T006_MSBClassLoader extends ClassLoader {
//重写findClass方法
@Override
    protected Class<?> findClass(String name) throws ClassNotFoundException { 
//定义指定的文件路径
        File f = new File("c:/test/", name.replace(".", "/").concat(".class"));
//使用字节数组输出流读取字节码文件 并保存在字节数组中
        try {
            FileInputStream fis = new FileInputStream(f);
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
            int b = 0;
            while ((b=fis.read()) !=0) {
                baos.write(b);
            }
            byte[] bytes = baos.toByteArray();
            baos.close();
            fis.close();//可以写的更加严谨
// 还是要依赖defineClass去依据具体路径,文件字节流,起止位置去做
            return defineClass(name, bytes, 0, bytes.length);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return super.findClass(name); //throws ClassNotFoundException
    }
    public static void main(String[] args) throws Exception {
        ClassLoader l = new T006_MSBClassLoader();
        Class clazz = l.loadClass("com.mashibing.jvm.Hello");
        Class clazz1 = l.loadClass("com.mashibing.jvm.Hello");
        System.out.println(clazz == clazz1);
        Hello h = (Hello)clazz.newInstance();
        h.m();

        System.out.println(l.getClass().getClassLoader());
        System.out.println(l.getParent());
        System.out.println(getSystemClassLoader());
    }
}

什么是懒加载?

需要使用该类时才去加载的一种策略。
但是却严格规定了初始化的时机:

  • new,getstatic,putstatic,invokestatic(访问静态变量)指令 访问final变量除外(访问final static memberVariable则不会加载)
  • java.lang.reflect对类进行反射调用
  • 初始化子类时,父类先初始化
  • 虚拟机启动时,被执行的主类必须初始化
  • 动态语言支持java.lang.invoke.MethodHandle解析的结果为REF_getstatic REF)putstatic REF_invokestatic的方法句柄时,该类必须初始化
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值