本地方法接口
本地方法
简单地讲,一个 Native Method 就是一个 Java 调用非 Java 代码的接口。一个 Native Method 是这样一个 Java 方法:该方法的实现由非 Java 语言实现,比如 C。这个特征并非 Java 所特有,很多其它的编程语言都有这一机制,比如在 C++中,你可以用 extern"c"告知 C++编译器去调用一个 C 的函数。
A native method is a Java method whose implementation is provided by non-java code
在定义ー个 native method 时,并不提供实现体(有些像定义ー个 Java interface),因为其实现体是由非 java语言在外面实现的。
本地接口的作用是融合不同的编程语言为Java所用,它的初衷是融合C/C++程序。
为什么使用NativeMethod
Java 使用起来非常方便,然而有些层次的任务用 Java 实现起来不容易,或者我们对程序的效率很在意时,问题就来了。
与 Java 环境外交互:
- 有时 Java 应用需要与 Java 外面的环境交互,这是本地方法存在的主要原因。你可以想想 Java 需要与一些底层系统,如操作系统或某些硬件交换信息时的情。本地方法正是这样一种交流机制:它为我们提供了一个非常简洁的接口,而且我们无需去了解 Java 应用之外的繁琐的细节。
与操作系统交互:
- JVM 支持着 Java 语言本身和运行时库,它是 Java 程序赖以生存的平台,它由一个解释器(解释字节码)和一些连接到本地代码的库组成。然而不管怎样,它毕竟不是一个完整的系统,它经常依赖于一些底层系统的支持。这些底层系统常常是强大的操作系统。通过使用本地方我们得以用 Java 实现了 jre 的与底层系统的交互,甚至 JVM 的一些部分就是用 c 写的。还有,如果我们要使用一些 Java 语言本身没有提供封装的操作系统的特性时,我们也需要使用本地方法。
Sun 's Java
- Sun 的解释器是用 C 实现的,这使得它能像一些普通的 C 一样与外部交互。jre 大部分是用 Java 实现的,它也通过一些本地方法与外界交互。例如:类 java.lang. Thread 的 setpriority()方法是用 Java 实现的,但是它实现调用的是该类里的本地方法 setpriority0()。这个本地方法是用 C 实现的,并被植入 JVM 内部,在 windows95 的平台上,这个本地方法最终将调用 Win32 Setpriority () API。这是一个本地方法的具体实现由 JVM 直接提供,更多的情况是本地方法由外部的动态链接库(external dynamic link library)提供,然后被 JVM 调用。
现状
目前该方法使用的越来越少了,除非是与硬件有关的应用,比如通过 Java 程序驱动打印机或者 Java 系统管理生产设备,在企业级应用中已经比较少见。因为现在的异构领域间的通信很发达,比如可以使用 Socket 通信,也可以使用 Web Service 等等,不多做介绍。
本地方法栈
Java虚拟机栈用于管理Java方法的调用,而本地方法栈用于管理本地方法的调用
本地方法栈,也是线程私有的
允许被实现成固定或者是可动态扩展的内存大小。(在内存溢出方面是相同的)
- 如果线程请求分配的容量超过本地方法栈允许的最大容量,Java 虚拟机将会抛出一个 StackOverflowError 异常。
- 如果本地方法可以动态扩展,并且在尝试扩展的时候无法申请到足够的内存,或者在创建新的线程时没有足够的内存去创建对应的本地方法栈,那么 Java 虚拟机将会抛出一个 outofmemoryerror 异常。
本地方法是使用C语言实现的
它的具体做法是 Native Method Stack 中登记 native 方法,在 Execution Engine 执行时加载本地方法库。
当某个线程调用一个本地方它就进入了一个全新的并且不再受虚拟机限制的世界。它和虚拟机拥同样的权限。
- 本地方法可以通过本地方法接口来访问虚拟机内部的运行时数据区
- 它甚至可以直接使用本地处理器中的寄存器
- 直接从本地内存的堆中分配任意数量的内存
并不是所有的 JVM 都支持本地方法。因为 Java 虚拟机规范并没有明确要求本地方法的使用语言、具体实现方式、数据结构等。如果 JVM 产品不打算支持 native 方法,也可以无需实现地方法栈
在 Hotspot JVM 中,直接将本地方法和虚拟机合二为一
堆
堆的核心概述
一个 JVM 实例只存在一个堆内存,堆也是 Java 内存管理的核心区域。
Java 堆区在 JVM 启动的时候即被创建,其空间大小也就确定了。是 JVM 管理的最大一块内存空间。堆内存的大小是可以调节的。
《Java 虚拟机规范》规定,堆可以处于物理上不连续的内存空间中,但在逻辑上它应该被视为连续的。
所有的线程共享 Java 堆,在这里还可以划分线程私有的缓冲区(Thread Local Allocation Buffer, TLAB)。
《Java虚拟机规范》中对Java堆的描述是:所有的对象实例以及数组都应当在运行时分配在堆上。
- "几乎"所有的对象实例都在这里分配内存。----从实际的角度看的
数组和对象可能永远不会存储在栈上,因为栈帧中保存引用,这个引用指向对象或者数组在堆中的位置。
在方法结束后,堆中的对象不会马上被移除,仅仅在垃圾回收时才会被移除。
堆,是GC(Garbage Collection)执行垃圾回收的重点区域。
现代垃圾收集器大部分都基于分代收集理论设计,堆空间细分为:
堆空间的内部结构(JDK7&JDK8)
设置堆内存大小与OOM
Java 堆区用于存储 Java 对象实例,那么堆的大小在 JVM 启动时就已经设定好了,大家可以通过选项"-Xmx"和"-Xms"来进行设置。
- "-Xms"用于表示堆区的起始内存,等价于-XX: InitialHeapsize
- "-Xmx"则用于表示堆区的最大内存,等价于-XX: MaxHeapsize
一旦堆区中的内存大小超过"-Xmx"所指定的最大内存时,将会抛出 OutOfMemoryError 异常。
通常会将-Xms 和 -Xmx 两个参数配置相同的值,其目的是为了能够在 java 垃圾回收机制清理完堆区后不需要重新分隔计算堆区的大小,从而提高性能。
默认情況下, 初始内存大小:物理电脑内存大小1/64
最大内存大小:物理电脑内存大小1/4
年轻代与老年代
存储在JVM中的Java对象可以被划分为两类:
- 一类是生命周期较短的瞬时对象,这类对象的创建和消亡都非常迅速
- 另一类对象的生命周期却非常长,在某些极端的情况下还能够与JVM生命周期保持一致。
Java 堆区进一步细分的话,可以划分为年轻代(YoungGen)和老年代(oldGen)其中年轻代又可以划分为 Eden 空间、Survivor0 空间和 Survivor1 空间(有时也叫做 from 区、to 区)。
新生代和老年代的比例一般为1:2,默认- XX:NewRatio=2,表示新生代占1,老年代占2,新生代占整个堆的1/3。
在 Hotspot 中,Eden 空间和另外两个 Survivor空间缺省所占的比例是 8:1:1 当然开发人员可以通过选项"-XX: SurvivorRatlo"调整这个空间比例。比如-XX: SurvivorRatio=8
几乎所有的 Java 对象都是在 Eden 区被 new 出来的。
绝大部分的 Java 对象的销毁都在新生代进行了。
- IBM 公司的专门研究表明,新生代中 80%的对象都是"朝生夕死"的
可以使用选项-Xmn"设置新生代最大内存大小
- 这个参数一般使用默认值就可以了。
图解对象分配过程
概述
- 为新对象分配内存是一件非常严谨和复杂的任务,JVM 的设计者们不仅需要考虑内存如何分配、在哪里分配等问题,并且由于内存分配算法与内存回收算法密切相关,所以还需要考虑 GC 执行完内存回收后是否会在内存空间中产生内存碎片。
- New 的对象先放伊甸园区。此区有大小限制。
- 当伊甸园的空间填满时,程序又需要创建对象,JVM 的垃圾回收器将对伊甸园区进行垃圾回收(Minor GC),将伊甸园区中的不再被其他对象所引用的对象进行销毁。再加载新的对象放到伊甸园区
- 然后将伊甸园中的剩余对象移动到幸存者0区
- 如果再次触发垃圾回收,此时上次幸存下来的放到幸存者 0 区的,如果没有回收,就会放到幸存者 1 区。
- 如果再次经历垃圾回收,此时会重新放回幸存者 0 区,接着再去幸存者 1 区。
- 啥时候能去养老区呢?可以设置次数。默认是 15 次。
- 可以设置参数:
--XX:MaxTenuringThreshold=<N>
进行设置
- 可以设置参数:
- 在养老区,相对悠闲。当养老区内存不足时,再次触发 GC: Major GC,进行养老区的内存清理
- 若养老区执行了 Maior GC 之后发现依然无法进行对象的保存,就会产生 OOM 异常
- Java.lang.OutOfMemoryError:Java heap space
总结
针对幸存者S0,S1区的总结:复制之后有交换,谁空谁是to。
关于垃圾回收:频繁发生在新生区,很少在养老区收集,几乎不在永久区/元空间收集。
Minor GC、Major GC、Full GC
JVM 在进行 GC 时,并非每次都对上面三个内存(新生代、老年代、方法区)区域一起回收的,大部分时候回收的都是指新生代。
针对 Hotspot VM 的实现,它里面的 GC 按照回收区域又分为两大种类型:一种是部分收集(Partial GC),一种是整堆收集(Full GC)
部分收集:不是完整收集整个Java 堆的垃圾收集。其中又分为
新生代收集(Minor GC/ Young GC):只是新生代的垃圾收集
老年代收集 (Major GC/Old GC):只是老年代的垃圾收集。
目前,只有 CMS GC 会有单独收集老年代的行为。
注意,很多时候 Major GC 会和 Full GC 混淆使用,需要具体分辨是老年代回收还是整堆回收。
混合收集(Mixed GC):收集整个新生代以及部分老年代的垃圾收集。
- 目前,只有 G1 GC 会有这种行为
整堆收集(Full GC):收集整个 java 堆和方法区的垃圾收集。
最简单的分代式GC策略的触发条件
年轻代GC(Minor GC)触发机制:
- 当年轻代空间不足时,就会触发 Minor GC,这里的年轻代满指的是 Eden 代满,Survivor 满不会引发 GC。(每次 Minor GC 会清理年轻代的内存。)
- 因为 Java 对象大多都具备朝生タ灭的特性,所以 Minor GC 非常频繁,一般回收速度也比较快。这一定义既清晰又易于理解
- Minor GC会引发 STW,暂停其它用户的线程,等垃圾回收结束,用户线程才恢复运行。
老年代GC(Major GC/Full GC)触发机制:
指发生在老年代的 GC,对象从老年代消失时,我们说"Major GC"或"Full GC"发生了。
出现了 Major Gc,经常会伴随至少ー次的 Minor GC(但非绝对的,在 Parallel Scavenge 收集器的收集策略里就有直接进行 Major GC 的策略选择过程)
- 也就是在老年代空间不足时,会先尝试触发 Minor GC。如果之后空间还不足则触发 Major Gc
Major GC 的速度一般会比 Minor GC慢10倍以上,STW 的时间更长。
如果 Major GC 后,内存还不足,就报 OOM 了。
Full GC触发机制:(后面详细介绍)
触发 Fu11GC 执行的情况有如下五种:
- 调用 System.gc()时,系统建议执行 Full GC,但是不必然执行
- 老年代空间不足
- 方法区空间不足
- 通过 Minor GC 后进入老年代的平均大小大于老年代的可用内存
- 由 Eden 区、survivor space0 (From Space)区向 survivor space1 (To Space)区复制时,对象大小大于 To Space 可用内存,则把该对象转存到老年代,且老年代的可用内存小于该对象大小
说明:Full GC 是开发或调优中尽量要避免的。这样暂时时间会短一些。
堆空间分代思想
为什么需要把Java堆分代?不分代就不能正常工作了吗?
经研究,不同对象的生命周期不同。70%-99%的对象都是临时对象。
新生代:有Eden、两块大小相同的Survivor构成,to总为空
老年代:存放新生代中经历多次GC仍然存活的对象
其实不分代完全可以,分代的唯一理由就是优化 GC 性能。如果没有分代,那所有的对象都在一块,就如同把一个学校的人都关在一个教室。GC 的时候要找到哪些对象没用这样就会对堆的所有区域进行扫描。而很多对象都是朝生夕死的,如果分代的话,把新创建的对象放到某一地方,当 GC 的时候先把这块存储"朝生夕死"对象的区域进行回收,这样就会腾出很大的空间出来。
内存分配策略
如果对象在 Eden 出生并经过第一次 Minor GC 后仍然存活,并且能被 Survivor 容纳的话,将被移动到 Survivor 空间中,并将对象年龄设为 1。对象在 Survivor 区中每熬过一次 MinorGC,年龄就增加 1 岁,当它的年龄增加到一定程度(默认为 15 岁,其实每个 JVM、每个 GC 都有所不同)时,就会被晋升到老年代中
对象晋升老年代的年龄阈值,可以通过选项-XX: MaxTenuringThreshold 来设置
针对不同年龄段的对象分配原则如下所示:
优先分配到 Eden
大对象直接分配到老年代
- 尽量避免程序中出现过多的大对象
长期存活的对象分配到老年代
动态对象年龄判断
- 如果 Survivor 区中相同年龄的所有对象大小的总和大于 Survivor 空间的一半,年龄大于或等于该年龄的对象可以直接进入老年代,无须等到 MaxTenuringThreshold 中要求的年龄。
空间分配担保
- -XX: HandlePromotionFailure
为对象分配内存:TLAB
为什么会有TLAB(Thread Local Allocation Buffer)?
- 堆区是线程共享区域,任何线程都可以访问到堆区中的共享数据
- 由于对象实例的创建在 JVM 中非常频繁,因此在并发环境下从堆区中划分内存空间是线程不安全的
- 为避免多个线程操作同一地址,需要使用加锁等机制,进而影响分配速度
什么是TLAB?
- 从内存模型而不是垃圾收集的角度,对 Eden 区域继续进行划分,JVM 为每个线程分配了一个私有缓存区域,它包含在 Eden 空间内。
- 多线程同时分配内存时,使用 TLAB 可以避免一系列的非线程安全问题,同时还能够提升内存分配的吞吐量,因此我们可以将这种内存分配方式称之为快速分配策略。
- 据我所知所有 OPEN JDK 衍生出来的 JVM 都提供了 TLAB 的设计。
TLAB再说明
- 尽管不是所有的对象实例都能够在 TLAB 中成功分配内存,但 JVM 确实是将TLAB 作为内存分配的首选
- 在程序中,开发人员可以通过选项"-XX: UseTLAB"设置是否开启TLAB 空间
- 默认情況下,TLAB 空间的内存非常小,仅占有整个 Eden 空间的 1%,当然我们可以通过选项"-XX: TLABWasteTargetPercent"设置 TLAB 空间所占用 Eden 空间的百分比大小。
- 一旦对象在 TLAB 空间分配内存失败时,JWM 就会尝试着通过使用加锁机制确保数据操作的原子性,从而直接在 Eden 空间中分配内存。
小结堆空间的参数设置
空间分配担保
在发生 Minor GC 之前,虚拟机会检查老年代最大可用的连续空间是否大于新生代所有对象的总空间。
如果大于,则此次 Minor GC 是安全的
如果小于,则虚拟机会查看-XX: HandlePromotionFailure 设置值是否允许担保失败。
如果 HandlePromotionFailure=true,那么会继续检査老年代最大可用连续空间是否大于历次晋升到老年代的对象的平均大小。
如果大于,则尝试进行一次 Minor GC,但这次 Minor GC 依然是有风险的
如果小于,则改为进行一次 Full GC。
如果 HandlePromotionFailure= false,则改为进行一次 Full GC
在 JDK6 Update24 之后,HandlePromotionFailure 参数不会再影响到虚拟机的空间分配担保策略,观察 OpenJDK 中的源码变化,虽然源码中还定义了HandlePromotionFailure 参数,但是在代码中已经不会再使用它。JDK6 Update 24 之后的规则变为只要老年代的连续空间大于新生代对象总大小或者历次晋升的平均大小就会进行 Minor GC,否则将进行 Full G。
堆是分配对象的唯一选择吗
在《深入理解 Java 虚拟机》中关于 Java 堆内存有这样一段描述:
随着 Java 编译期的发展与逃逸分析技术逐渐成熟,栈上分配、标量替换优化技术将会导致一些微妙的变化,所有的对象都分配到堆上也渐渐变得不那么"绝对"了。
在 Java 虚拟机中,对象是在 Java 堆中分配内存的,这是一个普遍的常识。但是,有种特殊情况,那就是如果经过逃逸分析(Escape Analysis)后发现,一个对象并没有逃逸出方法的话,那么就可能被优化成栈上分配。这样就无需在堆上分配内存,也无须进行垃圾回收了。这也是最常见的堆外存储技术。
此外,前面提到的基于 OpenJDK 深度定制的 TaoBao VM,其中创新的 GCIH (GC invisible heap)技术实现。off-heap,将生命周期较长的 Java 对象从 heap 中移至 heap 外,并且 GC 不能管理 GCIH 内部的 Java 对象,以此达到降低 GC 的回收频率和提升 GC 的回收效率的目的。
逃逸分析概述
如何将堆上的对象分配到栈,需要使用逃逸分析手段。
这是一种可以有效减少 Java 程序中同步负载和内存堆分配压力的跨函数全局数据流分析算法
通过逃逸分析,Java Hotspot 编译器能够分析出一个新的对象的引用的使用范围从而决定是否要将这个对象分配到堆上
逃逸分析的基本行为就是分析对象动态作用域:
当一个对象在方法中被定义后,对象只在方法内部使用,则认为没有发生逃逸。
当一个对象在方法中被定义后,它被外部方法所引用,则认为发生逃逸。例如作为调用参数传递到其他地方中。
- 没有发生逃逸的对象,则可以分配到栈上,随着方法执行的结束,栈空间就被移除。
net a porter:https://www.ikjzd.com/w/2132
代购公司:https://www.ikjzd.com/w/1982
为什么那么多人从日本回来都说,日本是一个去了还想再去的国家?:http://www.30bags.com/a/226036.html
为什么你身边的东北人感觉都很有钱?:http://www.30bags.com/a/245036.html
为什么清远那么多漂流点?:http://www.30bags.com/a/399680.html
为什么清真寺都有个洋葱顶|大象公会:http://www.30bags.com/a/231219.html
学长你的好大我不敢 乖都硬了让我进去疼你:http://lady.shaoqun.com/a/248329.html
男朋友把我抱到镜子前做 对着镜子从后面挺进她:http://lady.shaoqun.com/a/247489.html
深圳土星伴月时间是什么时候:http://www.30bags.com/a/517508.html
深圳大鹏新区红色主题公园有哪些:http://www.30bags.com/a/517509.html
深圳木星伴月是什么时候:http://www.30bags.com/a/517510.html
光明农场大观园停车场收费标准:http://www.30bags.com/a/517511.html
No comments:
Post a Comment