高级Java知识
高级Java知识(JVM、字节码、内存模型)
内存=方法区+栈空间+堆+程序计数器
栈(stack)包括虚拟机栈(VM stack)和本地方法栈(native method stack)。
方法区(method area)内放置 类信息(Class)、常量、静态变量、JIT编译后代码。
运行时常量池是方法区的一部分。
Class文件有一个常量池(Constant Pool Table),属于Class的一部分。
运行时常量池(Runtime Constant Pool)属于method area的一部分。
线程请求的栈容量大于VM允许的最大容量将引发StackOverflowError。
JVM规范允许虚拟机栈(如Hotspot虚拟机允许本地方法栈归入VM栈)动态扩展,若果在扩展时申请不到更多的内存,将引发OutOfMemoryError。
堆上申请不到更多内存将引发OOM error。
运行时常量池申请不到更多内存将引发OOM Error。
直接内存(Direct Memory)?也可能导致OOM。
new 过程: 寻找new 参数(需要被new的类)是否在常量池中、是否被加载、解析、初始化。
指针碰撞(bump the pointer)是认为内存是规整的,使用过的内存存在于(逻辑)连续的空间,用一个指针分割可用内存和已使用内存。
空闲列表(free list)是用列表记录空闲内存块。
对象头信息(object header)主要分两部分,一是类元数据指针(如果栈上reference通过句柄定位对象和类型数据,则没有类元指针),另一部分是对象hash、gc分代年龄、锁等自身运行时信息,如果是数组,还有数组大小信息。
分配内存、设置object header,初始化对象方法
对象在内存中分3块区域:object header, instance data, padding.
栈上reference定位对象有两种方式,一是句柄,二是直接指针。句柄是一种间接指针,该reference指向堆中句柄池中的一个句柄,句柄中存放对象实例数据指针和类型数据指针。直接指针则存放了包含类元数据指针的对象数据指针。
句柄定位方式优势在于对象在内存中被移动时(GC时被移动是很普遍的)栈上reference保持不变。直接指针优势是访问快。Sun HotSpot VM采用后者定位方式。
JVM参数:
- -Xss:栈大小
- -Xoss: native method stack,对Hotspot VM无效(其不区分vm stack、native method stack)
- -Xmx:堆最大内存量 -Xms:堆最小内存量
- -XX:PermSize=10M 永久代(位于方法区中)大小
- -XX:MaxPermSize=10M 永久代最大容量
jdk 1.7逐步去“永久代”,1.8完全去除。上述vm参数无效。
jvm运行时检测到的经常被抛出的jdk内置异常(NPE),可在其被抛出一定次数后优化其stacktrace为空,通过jvm开关OmitStackTraceInFastThrow设置。 - -XX:+OmitStackTraceInFastThrow 打开优化。
- -XX:-OmitStackTraceInFastThrow 关闭优化。
- -verbose:class -XX:+TraceClassLoading
- -XX:+TraceClassUnLoading关于类加载、卸载信息的vm参数。
运行时JVM会对经常被抛出的JRE预定义异常进行输出优化——将其stacktrace置为空,减少性能损失,所以可能看到一个异常(如NPE)被抛出,但其堆栈信息为空。
引用计数算法无法解决循环引用问题,objA.x=objB; objB.y=objA;objA=null;objB=null; 由于相互引用导致两个计数都不为0,但实际两个都应该被GC。
可达性分析GC策略:能通过GC roots可达的对象是存活对象。
GC Roots包括以下对象:
- VM栈(栈帧中本地变量表)中的引用对象
- 方法区中静态对象
- native方法中引用的对象
引用:强引用(Strong)、软引用(soft)、弱引用(weak)、虚引用(phantom)。强度依次减弱。
软引用:还有用但并非必需。在内存不够用(将要溢出前)才会回收这些引用。
弱引用:弱引用不影响GC,但对象只有弱引用时会被正常GC。
虚引用:完全不影响GC,唯一目的是在此对象被GC时能收到一条系统通知。
GC:GC roots不可达且有必要执行finalize()方法(对象重写了finalize且没有被vm调用过) -> 第一次标记,并放到F-Quene -> 低优先级执行F-Quene中对象的finalize(但不保证执行完该方法因其运行慢或死循环) -> finalize期间对象没有将this指向其他gc roots可达对象的字段:回收;否则不回收。
注:任一对象的finalize只被执行一次,即使再次面对GC。遵从建议:让我们忘了这个方法吧。
方法区的回收是针对无用常量和类。
GC算法:
- Mark-sweep 标记清除。标记需要回收的对象,标记完后同意清除。不足:标记和清除效率都不高;清除后产生大量零碎、不连续内存空间。
- copying 复制。 Eden空间+两块Survivor空间。Eden:Survivor=8:1,回收eden,将其中存活对象复制到survivor空间,清除eden。缺点:如果对象存活率高,则导致过多复制而降低gc性能;由于10%的survivor空间不能保证一定能容纳所有存活对象,需要额外空间保证。
- mark-compact 标记-整理。老年代中对象存活率很高,复制算法导致过多复制以及额外空间。标记-整理算法在gc时将存活对象整理到内存空间的一端连续存放,直接清理另一端内存。
- Generational Collection 分代收集。将内存分为新生代和老年代,根据特点使用copying、mark-sweep/mark-compact算法。
GC器:
- serial。 单线程,新生代用copying、老年代用mark-compact算法,准备收集时会“stop the world”。
- ParNew。serial的多线程版
- Parallel Scavenge。新生代收集器。关注点是可控的吞吐量(user-time / (user-time + gc-time)),而CMS等GC器关注GC时导致的停顿时间。该GC器可自适应调节新生代大小(—Xmn)、eden与survivor比例(_XX:SurvivorRatio)、晋升老年代年龄(-XX:PretenureSizeThreshold)。
- Serial Old。Serial的老年代版本。
- Parallel Old。Parallel Scavenge的老年代版本,使用mark-compact算法。
- CMS(Concurrent Mark Sweep)。并发GC、低停顿时间。 a. 初始标记(gc roots直接关联的对象,stop the world) b. 并发标记(gc roots tracing,与用户线程同时工作) c. 重新标记(并发标记期间产生的变动,stop the world) d. 并发清除。
- G1(Garbage First)。优点:a. 并行、并发;b. 分代收集;c. 内存碎片整理;d. 可控的停顿时间。思想:内存分为大小相同的若干region,优先回收价值(获得的内存空间/所需时间)大的garbage。
JVM工具
jps -l
显示jvm进程信息
jstack
[-F 强制输出堆栈; -l 锁信息; -m native方法堆栈] pid
VisualVM(jvisualvm): btrace插件,热机跟踪。jvisualvm在jdk9+后不再默认提供。
(hotspot)JVM引擎会复用局部变量表slot,对GC影响的例子如下:
// VM参数: -verbose:gc
void main(){
{
byte[] mem=new byte[64*1024*1024];
}
// int a=0; 无此代码行时尽管mem已不能被访问,但不会被回收,当启用此代码时由于再次访问了局部变量表,将导致$mem的内存被回收
System.gc();
}
方法正常完成退出、方法异常完成退出。后者需要异常处理器表帮助方法返回、继续执行程序。
字节码指令集
invokestatic: 调用静态方法
invokespecial: , private, 父类方法。
invokevirtual: 虚方法(除被invokestatic, invokespecial指令调用的方法即static, , private, 父类方法的方法)及final方法。
invokeinterface:
invokedynamic:
非虚方法(static、、private、父类方法、final方法)在类加载时即可被解析为直接引用。final非虚方法由invokevirtual指令调用。
java是静态多分派、动态单分派的语言。
类加载
编译器会为类生成方法,接口的只有在常量初始化时才会被生成。
类执行前一定先执行父类,不会执行类接口的。
类只会被初始化一次,类中(静态初始化过程中)应避免过长的执行时间,否则会导致阻塞与此类有关其他线程。
类的Class依赖于类加载器,不同类加载器加载同一份类字节码得到的Class一定不一样。两个类的“相等”比较只有在类被同一个加载器加载的情况下有意义(否则一定不相等),这里的相等包括,Class<?>.equals, isAssignableFrom, inInstance, instanceof。
boostrap classloader <- ext classloader <- application classloader(system classloader) <- user classloader
高级Java知识的更多相关文章
- [原创]上海好买基金招高级Java技术经理/运维主管/高级无线客户端开发等职位(内推)
[原创]上海好买基金招高级Java技术经理/运维主管/高级无线客户端开发等职位(内推) 内部推荐职位 高级JAVA技术经理: 岗位职责: 负责项目管理(技术方向),按照产品开发流 ,带领研发团队,制定 ...
- 如何成为高级java程序员
或许您已经读过我的那篇小文<如何成为java初级程序员>,那里面只介绍了成为一个JAVA程序员应该具备的一些知识.我相信您绝不会只想着做一个初级的程序员,上了软件开发的小船,您肯定有着远大 ...
- Java知识回顾 (1) 编译环境与基本变量类型
参考资料 runoob Java知识回顾序列的相关资料,主要来自 runoob,并对其中的知识进行概况或总结,去除对一个之前了解过Java的人员无关的知识点.以便能够使得一个新手,或之前有Java经验 ...
- 互联网高级Java面试总结
前不久刚换了单位,这段时间抽出时间来总结一下. 本人渣本毕业四年,无大厂工作经验,出来面高级Java. 上家单位是一个知名互联网平台,但是体量不大的小公司(5线互联网公司),但就是出名(职场人都知道~ ...
- 高级Java必看的10本书
1.深入理解Java虚拟机:JVM高级特性与最佳实践 本书共分为五大部分,围绕内存管理.执行子系统.程序编译与优化.高效并发等核心主题对JVM进行了全面而深入的分析,深刻揭示了JVM的工作原理. 2. ...
- Java知识体系
Java知识体系 java知识结构.jpg web框架.jpg 计算机课程体系.png 2016-08-19_090929.png 流行的哈希算法生存状况.jpg "JAVA之父" ...
- [转]20个高级Java面试题汇总
http://saebbs.com/forum.php?mod=viewthread&tid=37567&page=1&extra= 这是一个高级Java面试系列题中的第一部分 ...
- 20个高级Java面试题
这是一个高级Java面试系列题中的第一部分.这一部分论述了可变参数,断言,垃圾回收,初始化器,令牌化,日期,日历等等Java核心问题. 程序员面试指南:https://www.youtube.com/ ...
- 高级java高并发,高性能,分布式,高可用,负载均衡,系统架构实战
java架构师.集群.高可用.高可扩 展.高性能.高并发.性能优化.Spring boot.Redis.ActiveMQ.Nginx.Mycat.Netty.Jvm大型分布 式项目实战 视频课程包含: ...
随机推荐
- HBase连接数据库(集群)
一.使用java接口对hbase进行表的创建1.引入需要的jar包2.代码: public static void main(String[] args) throws Exception { //得 ...
- 1067 Bash游戏 V2
有一堆石子共有N个.A B两个人轮流拿,A先拿.每次只能拿1,3,4颗,拿到最后1颗石子的人获胜.假设A B都非常聪明,拿石子的过程中不会出现失误.给出N,问最后谁能赢得比赛. 例如N = 2.A只能 ...
- TDSTCPServerTransport 的Filters
TDSTCPServerTransport 的Filters TDSTCPServerTransport 的 Filter 属性,可以对传递的数据进行加密,压缩,再修改等,有 点注入的概念.默认情况下 ...
- Android学习路线(十八)支持不同设备——支持不同的屏幕
Android系统使用两个普通属性:尺寸和密度,来对设备屏幕进行分类. 你须要先预測你的应用将会在什么样屏幕的设备上安装,包含屏幕尺寸和密度.这种话,你就须要提供一些可选的资源类让你的应用在不同屏幕的 ...
- 5.配置globals文件(目标端)
mgr进程是goldengate软件执行的主进程.是由这个进程控制其它进程的,比方extract,replicat进程等. 对于mgr进程的配置,将会在以下介绍. global文件我们 ...
- es6 Object.assign ECMAScript 6 笔记(六) ECMAScript 6 笔记(一) react入门——慕课网笔记 jquery中动态新增的元素节点无法触发事件解决办法 响应式图像 弹窗细节 微信浏览器——返回操作 Float 的那些事 Flex布局 HTML5 data-* 自定义属性 参数传递的四种形式
es6 Object.assign 目录 一.基本用法 二.用途 1. 为对象添加属性 2. 为对象添加方法 3. 克隆对象 4. 合并多个对象 5. 为属性指定默认值 三.浏览器支持 ES6 O ...
- POJ 1300 Door Man(欧拉通路)
题目描写叙述: 你是一座大庄园的管家. 庄园有非常多房间,编号为 0.1.2.3..... 你的主人是一个心不在 焉的人,常常沿着走廊任意地把房间的门打开.多年来,你掌握了一个诀窍:沿着一个通道,穿 ...
- 阿里大数据比赛sesson2_RF&GBRT(下)
-----------__-----------接上文---------__---------- 2.Xlab RF上手 2.1.训练特征表准备 训练的特征表gbrt_offline_section_ ...
- apt --fix-broken install
1 自动修复安装出现broken的package 但是,如果还是失败的话,就需要手动进行干预了.
- javascript设置和获取cookie的方法
设置cookie的方法,和获取cookie的方法例如以下 设置cookie document.cookie="name="+value; //获取cookie当中index是coo ...