java⾃学百度⽹盘,绝对⼲货分享
⾯:为什么要使⽤双亲委派机制去加载类?
答:避免多份同样字节码的加载,浪费内存。
类的加载⽅式
隐式加载:new
显⽰加载:loadClass、forName等
类的装载过程如下图:
⾯:loadClass和forName的区别?
Class.forName得到的class是已经初始化完成了的 (MySQL加载驱动时,需要调⽤静态代码块完成⼀些操作)ClassLoader.loadClass得到的class是还没有链接的。(⽤于Spring IoC中的延迟加载机制)
Java内存模型
JVM内存模型——JDK8如下图所⽰:
线程私有:程序计数器、虚拟机栈、本地⽅法栈
线程共享:MetaSpace、Java堆
程序计数器(PC)
当前线程所执⾏的字节码⾏号指⽰器(逻辑)
通过改变计数器的值来选取下⼀条需要执⾏的字节码指令
和线程是⼀对⼀的关系即“线程私有”
对Java⽅法计数,如果是Native⽅法则计数器的值为Undefined
不会发⽣内存泄漏
Java虚拟机栈(Stack)
Java⽅法执⾏的内存模型
包含多个栈帧(⼀个栈帧包括局部变量表、操作栈、动态链接、返回地址等,⽅法的调⽤即对于栈帧从虚拟机Stack中⼊栈到出栈的过程)
当线程请求的栈深度超过最⼤值,会抛出 StackOverflowError 异常;
栈进⾏动态扩展时如果⽆法申请到⾜够内存,会抛出 OutOfMemoryError 异常。
本地⽅法栈
与虚拟机栈相似,主要作⽤于标注了native的⽅法。
元空间(MetaSpace)
⽤于存放已被加载的类信息、常量、静态变量。
⾯:谈谈元空间(MetaSpace)和永久代(PermGen)的区别?
均是⽅法区(JVM的⼀种规范)的实现
JDK8后元空间替代了永久代
元空间使⽤本地内存,⽽永久代使⽤的是jvm内存,这解决了空间不⾜的问题。
⾯:MetaSpace相⽐PermGen的优势?
字符串常量池(JDK1.7开始移动到Java堆中)存在与永久代中,容易出现性能问题和内存溢出
类的⽅法的信息⼤⼩难以确定,给永久代的⼤⼩的指定带来了困难
永久代会为GC带来不必要的复杂性
⽅便HotSpot与其他JVM如Jrockit的集成(因为永久代是HotSpot独有的)
Java堆(Heap)
是对象实例的分配区域
GC管理的主要区域
⾯:说说JVM三⼤性能调优参数-Xms、-Xmx、-Xss的含义?
java -Xms128m -Xmx128m -Xss256k -jar xxx.jar
-Xss:规定了每个线程虚拟机栈(堆栈)的⼤⼩
-Xms:堆的初始值
-Xmx:堆能达到的最⼤值
⼀般将-Xms与-Xmx设置为同样的数值,避免堆扩容时发⽣的内存抖动,影响程序的稳定性。
内存分配策略
静态存储:编译时确定每个数据⽬标在运⾏时的存储空间需求
栈式存储:数据区需求在编译时未知,在运⾏时模块⼊⼝前确定
堆式存储:编译时或运⾏时模块⼊⼝都⽆法确定需求,需要动态分配
⾯:谈谈Java内存模型中堆和栈的区别与联系?
联系:引⽤对象、数组时,栈⾥定义变量来保存堆中⽬标的⾸地址
管理⽅式:栈⾃动释放,堆需要GC
空间⼤⼩:⼀般栈⽐堆⼩
碎⽚相关:栈产⽣的内存碎⽚远⼩于堆
分配⽅式:栈⽀持静态和动态分配,⽽堆仅⽀持动态分配
效率:栈的效率⽐堆⾼(栈只有⼊栈与出栈)
⾯:请解释下JDK6和JDK6+下intern()⽅法的区别?
JDK6:当调⽤intern⽅法时,如果字符串常量池先前已创建出该字符串对象,则返回池中的该字符串的引⽤。否则将此字符串对象添加到字符串常量池中,并且返回该字符串对象的引⽤。
java在哪里可以免费自学JDK6+:当调⽤intern⽅法时,如果字符串常量池先前已创建出该字符串对象,则返回池中该字符串的引⽤。否则,如果该字符串对象已经存在与Java堆中,则将堆中对此对象的引⽤添加到字符串常量池中,并且返回该引⽤;如果堆中不存在该对象,则在字符串常量池中创建该字符串并返回其引⽤。
最后
由于篇幅限制,⼩编在此截出⼏张知识讲解的图解,有需要的程序猿(媛)可以点赞后获取哦
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论