目录
5. 反射中,Class.forName() 和 ClassLoader.loadClass()区别
5. 内存泄露和内存溢出分别是什么?什么原因造成?如何避免?
Java内存模型
Java虚拟机中共有五大内存区域
程序计数器,Java虚拟机栈,本地方法栈,java堆,方法区
程序计数器:是一块很小的内存空间,用于存储当前线程执行字节码文件的行号指示器。
虚拟机栈:是描述Java方法执行过程的内存模型,每个方法被执行时都会创建一个栈桢,栈桢中存储了局部变量表,操作数栈,动态链接,方法出口等。
本地方法栈:和虚拟机栈作用类似,区别是虚拟机栈为Java方法服务,本地方法栈为Native方法服务。
Java堆:JVM运行过程中创建的对象和生成的数据都存储在堆中,堆是被线程共享的内存区域,也是垃圾回收最主要的内存区域。
方法区:用来存储常量、静态变量、类信息、即时编译器编译后的机器码、运行时常量池等数据。
2.java内存模型中堆和栈的区别
联系:引用对象、数组时,栈里定义变量保存堆中目标的首地址
区别:
管理方式:栈自动释放,堆需要GC
空间大小:栈比堆小
碎片相关:栈产生的碎片远小于堆
分配方式:栈支持静态和动态分配,而堆仅支持动态分配
效率:栈的效率比堆高
3.类的装载过程(类加载机制)
(1)加载:把class文件加载到内存中
(2)链接(又分为验证、准备、解析)
1、验证:检查加载到内存的文件是否符合格式要求
2、准备:为类变量(static修饰的变量)赋初值(系统默认值)
3、解析:把符号引用转换为直接引用
(3)初始化:执行类变量赋值和静态代码块
4.谈谈类加载器的双亲委派机制
类加载器之间的层次关系叫做双亲委派模型,要求除了顶层的启动类加载器外,其余的类加载器都应当有自己的父类加载器。
一个类收到类加载请求后会层层找父类加载器去尝试加载,因此所有的加载请求最终都会被传送到顶层的启动类加载器,只有当父类加载器反馈自己无法完成加载时,子类加载器才会尝试自己去加载。
好处:避免同样字节码多次重复加载(内存宝贵,避免保存重复的class类对象)
实现:先判断该类是否被加载过,如果未被加载就调用父类的 loadClass()方法,如果父类为null,则使用启动类加载器作为加载器。如果父类加载失败,抛出 notfound 异常,则调用自己的 findClass 进行加载。简单概括:自底向上检查类是否被加载,自顶向下尝试加载类。
5. 反射中,Class.forName() 和 ClassLoader.loadClass()区别
➢forName得到的class是已经初始化完成的,它会执行static代码
创建数据库驱动:Class. forName (”com. mysql.jdbc. Driver" )
➢loadClass得到的class是还没有链接的