java中+=是什么意思Garbage Collector MXBean的意思是

     最近测试环境总是报OOM的异常正恏赶上毕玄的JVM培训,于是花了一天的时间去参加培训培训后,对于JVM内存的管理有了初步的印象最起码不那么疑惑了,之后又找了两本書细化的看了一下现在总结记录下来。

程序计数器:是一块较小的内存空间作用可以看做是当前线程所执行的字节码的行号的指示器,线程私有

JVM方法栈和本地方法栈:在sun的jdk中,JVM方法栈和本地方法栈是算在一起的虚拟机栈为虚拟机执行java方法,本地方法栈为虚拟机执行Native方法线程私有。

java heap:是虚拟机中内存区域最大的一块细分可以分为新生代和旧生带,新生代可以划分为E、S0、S1其中S1和S0又可以叫做from 或 to,线程共享

方法区:存放了要加载的类的信息,类中的静态变量定义为final的常量,类中的Field信息方法信息等,全局共享又叫做持久带,可鉯通过  -XX:PermSize和-XX:MaxPermSize设置最小值和最大值线程共享。

    大多数情况下java中+=是什么意思新建的对象都是在新生代上分配的,新生代由Eden和两块相同大小的S0囷S1组成其中S0和S1又称为From和To(这个划分没有先后顺序),可以通过-Xmn来设置新生代的大小-XX:SurvivorRatio设置Eden和S区的比值,有些垃圾回收器会对S0或者S1进行動态的调整

     之所以说大多数情况下新建的对象在新生代上分配,是因为有两种情况下java新创建的对象会直接到旧生带一种是大的数组对潒,且对象中无外部引用的对象另外一种是通过

启动参数上面进行设置-XX:PretenureSizeThreshold=1024(单位是字节),意思是对象超过此大小就直接分配到旧生带生媔。此外并行垃圾回收器可以在运行期决定那些对象可以直接创建在旧生带。

多次回收之后仍然存活的对象大小是-Xms减去-Xmn。

从主机读取數据的超时时间

1、旧生代空间只有在新生代对象转入及创建为大对象、大数组时才会出现不足的现象当执行Full GC后空间仍然不足,则抛出如丅错误:

2、Permanet Generation中存放的为一些class的信息等当系统中要加载的类、反射的类和调用的方法较多时,Permanet Generation可能会被占满在未配置为采用CMS GC的情况下会執行Full GC。如果经过Full GC仍然回收不了那么JVM会抛出如下错误信息:

4、线程请求的栈深度大于虚拟机所允许的最大深度

大体对于JVM内存管理有了一个叻解,后续碰到相关的问题后再进行更加深入的了解

在网上看见一篇介绍jvm启动参数的文章,搞下来呵呵

本文是基于最新的SUN官方文档 编写嘚译文主要介绍JVM中的非稳态选项及其使用说明。

为了让读者明白每个选项的含义作者在原文基础上补充了大量的资料。希望这份文档对正在研究JVM参数的朋友有帮助!

另外,考虑到本文档是初稿如有描述错误,敬请指正

允许为java进程安装信号处理器。

Java信号处理相关知識详见 

注意:你熟悉的代码里没调用System.gc(),不代表你依赖的框架工具没在使用

例如RMI就在多数用户毫不知情的情况下,显示地调用GC来防止自身OOM

请仔细权衡禁用带来的影响。

在Class校验器中放松对访问控制的检查。

启用CMS低停顿垃圾收集器

其他情况下,默认不启用

策略为新生代使用并行清除年老代使用单线程Mark-Sweep-Compact的垃圾收集器。

策略为老年代和新生代都使用并行清除的垃圾收集器

其他情况下,默认不启用

使用新嘚Class类型校验器

新Class类型校验器有什么特点? 
新Class类型校验器将老的校验步骤拆分成了两步: 
新类型校验器通过在javac编译时嵌入类型信息到bytecode中,省略了类型推断这一步从而提升了classloader的性能。

Java6新引入选项默认启用

如果新的Class校验器检查失败,则使用老的校验器

因为JDK6最高向下兼容箌JDK1.2,而JDK1.2的class info 与JDK6的info存在较大的差异所以新校验器可能会出现校验失败的情况。

java5以前是默认不启用java6默认启用

为了确保minor gc能够顺利完成,GC需要在姩老代中额外保留一块足以容纳所有活跃对象的内存空间 
为什么要关闭新生代收集担保? 
因为在年老代中预留的空间大小是无法精确計算的。

为了确保极端情况的发生GC参考了最坏情况下的新生代内存占用,即Eden+First Survivor

这种策略无疑是在浪费年老代内存,从时序角度看还会提前触发Full GC。

为了避免如上情况的发生JVM允许开发者手动关闭新生代收集担保。

在开启本选项后minor gc将不再提供新生代收集担保,而是在出现survior戓年老代不够用时抛出promotion failed异常。

启用多线程自旋锁优化

大家知道,Java的多线程安全是基于Lock机制实现的而Lock的性能往往不如人意。 
互斥是一種会导致线程挂起并在较短的时间内又必须重新调度回原线程的,较为消耗资源的操作

为了避免进入OS互斥,Java6的开发者们提出了自旋锁優化

自旋锁优化的原理是在线程进入OS互斥前,通过CAS自旋一定的次数来检测锁的释放

如果在自旋次数未达到预设值前锁已被释放,则当湔线程会立即持有该锁

-XX:+UseSpinning 必须先启用,对于java6来说已经默认启用了这里默认自旋10次

控制多线程自旋锁优化的自旋次数。(什么是自旋锁优化见 -XX:+UseSpinning 处的描述)

限制GC的运行时间。如果GC耗时过长就抛OOM。

1.4.2以前和使用-client选项时默认不启用,其余版本默认启用

使用本地线程的优先级

为了防止与其他发送信号的应用程序冲突,允许使用候补信号替代 SIGUSR1和SIGUSR2

绑定所有的用户线程到内核线程。 
减少线程进入饥饿状态(得不到任何cpu time)的次数

使用轻量级进程(内核线程)替换线程同步。

设置java进程可用文件描述符为操作系统允许的最大值

在solaris中,允许运行时中断线程


启用JVM开发团队最新的调优成果。例如编译优化偏向锁,并行年老代收集等

通过JIT编译器,将方法编译成机器码的触发阀值可以理解為调用方法的次数,例如调1000次将方法编译为机器码。

设置堆内存的内存页大小

调整内存页的方法和性能提升原理,详见 

GC后如果发现涳闲堆内存占到整个预估堆内存的70%,则收缩堆内存预估最大值

预估堆内存是堆大小动态调控的重要选项之一。

堆内存预估最大值一定小於或等于固定最大值(-Xmx指定的数值)

前者会根据使用情况动态增大或缩小,以提高GC回收的效率

新生代占整个堆内存的最大值。

Perm(俗称方法區)占整个堆内存的最大值

GC后,如果发现空闲堆内存占到整个预估堆内存的40%则增大堆内存的预估最大值。此值不会超过固定最大值

噺生代和年老代的堆内存占用比例。

例如2表示新生代占最大堆内存的1/2即年老代和新生代平分堆的占用。

新生代预估堆内存占用的默认值(什么是预估堆内存?见 -XX:MaxHeapFreeRatio 处的描述)

设置代码缓存的最大值编译时用。

实际使用的survivor空间大小占比默认是50%,最高90%

偏向锁原理详见 

优化原始类型的getter方法性能。

调整内存页的方法和性能提升原理详见

与机器码指令预读相关的一个选项,资料比较少本文档不做解释。有兴趣嘚朋友请自行阅读官方doc

与机器码指令预读相关的一个选项,资料比较少本文档不做解释。有兴趣的朋友请自行阅读官方doc


打印JIT编译器編译耗时。

如果JVM crashed将错误日志输出到指定文件路径。

堆内存快照的存储文件路径

当java进程因OOM或crash被OS强制终止后,会生成一个hprof(Heap PROFling)格式的堆内存快照文件该文件用于线下调试,诊断查找问题。

在OOM时输出一个dump.core文件,记录当时的堆内存快照(什么是堆内存快照? 见 -XX:HeapDumpPath 处的描述)

當java每抛出一个ERROR时,运行指定命令行指令集指令集是与OS环境相关的,在linux下多数是bash脚本windows下是dos批处理。

当第一次发生OOM时运行指定命令行指囹集。指令集是与OS环境相关的在linux下多数是bash脚本,windows下是dos批处理

5.0 引入,默认不启用

往stdout打印方法被JIT编译时的信息

1.4.0引入,默认不启用

打印对潒的存活期限信息

Age1 2表示在第1和2次GC后存活的对象大小。

1.4.2引入默认不启用

1.4.2引入,默认不启用

打印所有静态类常量的代码引用位置。用于debug

Java6 引入,默认不启用

打印class的装载策略变化信息到stdout

装载策略变化是实现classloader隔离/名称空间一致性的关键技术。

对此感兴趣的朋友详见 中的

当java進程因OOM或crashed被强制终止后,生成一个堆快照文件(什么是堆内存快照? 见 -XX:HeapDumpPath 处的描述)

允许为java进程安装信号处理器。


Java信号处理相关知识详见 

紸意:你熟悉的代码里没调用System.gc(),不代表你依赖的框架工具没在使用

例如RMI就在多数用户毫不知情的情况下,显示地调用GC来防止自身OOM

请仔細权衡禁用带来的影响。

在Class校验器中放松对访问控制的检查。

启用CMS低停顿垃圾收集器

其他情况下,默认不启用

策略为新生代使用并行清除年老代使用单线程Mark-Sweep-Compact的垃圾收集器。

策略为老年代和新生代都使用并行清除的垃圾收集器

其他情况下,默认不启用

新Class类型校验器有什么特点新Class类型校验器,将老的校验步骤拆分成了两步:


新类型校验器通过在javac编译时嵌入类型信息到bytecode中省略了类型推断这一步,从而提升了classloader的性能

Java6新引入选项,默认启用

如果新的Class校验器检查失败则使用老的校验器。

因为JDK6最高向下兼容到JDK1.2而JDK1.2的class info 与JDK6的info存在较大的差异,所以新校验器可能会出现校验失败的情况

java5以前是默认不启用,java6默认启用


然而Second Survivor不一定能容纳下所有从E和F区copy过来的活跃对象。

为了确保minor gc能夠顺利完成GC需要在年老代中额外保留一块足以容纳所有活跃对象的内存空间。

为什么要关闭新生代收集担保因为在年老代中预留的空間大小,是无法精确计算的

为了确保极端情况的发生,GC参考了最坏情况下的新生代内存占用即Eden+First Survivor。

这种策略无疑是在浪费年老代内存從时序角度看,还会提前触发Full GC

为了避免如上情况的发生,JVM允许开发者手动关闭新生代收集担保

在开启本选项后,minor gc将不再提供新生代收集担保而是在出现survior或年老代不够用时,抛出promotion failed异常

启用多线程自旋锁优化。

大家知道Java的多线程安全是基于Lock机制实现的,而Lock的性能往往鈈如人意
互斥是一种会导致线程挂起,并在较短的时间内又必须重新调度回原线程的较为消耗资源的操作。

为了避免进入OS互斥Java6的开發者们提出了自旋锁优化。

自旋锁优化的原理是在线程进入OS互斥前通过CAS自旋一定的次数来检测锁的释放。

如果在自旋次数未达到预设值湔锁已被释放则当前线程会立即持有该锁。

限制GC的运行时间如果GC耗时过长,就抛OOM

1.4.2以前和使用-client选项时,默认不启用其余版本默认启鼡

使用本地线程的优先级。

为了防止与其他发送信号的应用程序冲突允许使用候补信号替代 SIGUSR1和SIGUSR2。

绑定所有的用户线程到内核线程
减少線程进入饥饿状态(得不到任何cpu time)的次数。

使用轻量级进程(内核线程)替换线程同步

设置java进程可用文件描述符为操作系统允许的最大徝。

在solaris中允许运行时中断线程 。

启用JVM开发团队最新的调优成果例如编译优化,偏向锁并行年老代收集等。

通过JIT编译器将方法编译荿机器码的触发阀值,可以理解为调用方法的次数例如调1000次,将方法编译为机器码

设置堆内存的内存页大小。

调整内存页的方法和性能提升原理详见 

GC后,如果发现空闲堆内存占到整个预估上限值的70%则收缩预估上限值。

JVM在启动时会申请最大值(-Xmx指定的数值)的地址涳间,但其中绝大部分空间不会被立即分配(virtual)

它们会一直保留着,直到运行过程中JVM发现实际占用接近已分配上限值时,才从virtual里再分配掉┅部分内存

这里提到的已分配上限值,也可以叫做预估上限值


引入预估上限值的好处是,可以有效地控制堆的大小堆越小,GC效率越高嘛

注意:预估上限值的大小一定小于或等于最大值。

新生代占整个堆内存的最大值

Perm(俗称方法区)占整个堆内存的最大值。

GC后如果发现空闲堆内存占到整个预估上限值的40%,则增大上限值

新生代和年老代的堆内存占用比例。

例如2例如2表示新生代占年老代的1/2占整个堆内存的1/3。

设置代码缓存的最大值编译时用。

因为我们的新生代有2个survivor即S1和S22。所以survivor总共是占用新生代内存的 2/10Eden与新生代的占比则为 8/10。

实際使用的survivor空间大小占比默认是50%,最高90%

偏向锁原理详见 

优化原始类型的getter方法性能。

调整内存页的方法和性能提升原理详见

与机器码指囹预读相关的一个选项,资料比较少本文档不做解释。有兴趣的朋友请自行阅读官方doc

与机器码指令预读相关的一个选项,资料比较少本文档不做解释。有兴趣的朋友请自行阅读官方doc

打印JIT编译器编译耗时。

如果JVM crashed将错误日志输出到指定文件路径。

堆内存快照的存储文件路径

当java进程因OOM或crash被OS强制终止后,会生成一个hprof(Heap PROFling)格式的堆内存快照文件该文件用于线下调试,诊断查找问题。

当java每抛出一个ERROR时運行指定命令行指令集。指令集是与OS环境相关的在linux下多数是bash脚本,windows下是dos批处理

当第一次发生OOM时,运行指定命令行指令集指令集是与OS環境相关的,在linux下多数是bash脚本windows下是dos批处理。

往stdout打印方法被JIT编译时的信息

1.4.0引入,默认不启用

打印对象的存活期限信息

Age1 2表示在第1和2次GC后存活的对象大小。

1.4.2引入默认不启用

1.4.2引入,默认不启用

打印所有静态类常量的代码引用位置。用于debug

打印class的装载策略变化信息到stdout。

装载筞略变化是实现classloader隔离/名称空间一致性的关键技术

完善的单元测试,功能回归测试和性能基准测试可以减少因调整非稳态JVM选项带来的风險。

Java6性能调优白皮书

}

大概意思好像说JVM虚拟机的环境 1.4版夲过低用1.5版本以上的,很疑惑。安装

}
  • 定义:注解(Annotation)也叫元数据,昰一种代码级别的说明是Java 的JDK1.5版本开始引入的一个特性,与类、接口、枚举是在同一个层次它可以声明在包、类、属性、方法、局部变量、方法参数等的前面,用来对这些元素进行说明注释。
  • 作用分类:  很多框架技术的新版本中都可以使用注解替代XML配置文件;
    • ①编写攵档:通过代码里标识的元数据生成文档【生成文档doc文档】
    • ② 代码分析:通过代码里标识的元数据对代码进行分析【使用反射】
    • ③编译检查:通过代码里标识的元数据让编译器能够实现基本的编译检查【Override】
  • 运行时的Java实例对象存储在堆内存空间中。当一个对象不再被引用了咜变成可被从堆内存中回收空间。在垃圾回收的过程中这些对象将被从堆内存中清除,同时它们的空间也就被回收了 
  • 堆内存的空间根據对象的存活时限主要分成了三部分:年轻代、老年代、永久代
    • a-Eden区(所有实例在运行时最初都分配到eden区中)
    • c,-S1 Survivor Space(再老一些的对象被从S0区移动到S1区,其实是在Young GC过程中S0区已满则会将eden区中还存活的对象和S0区中的存活对象移动到S1区中
    • 经过S0,S1中几轮迭代后还存活的对象被提升到老年代 
    • 包含一些元数据像类、方法等等永久代空间在JDK8特性中已经被移除
  • Java垃圾回收是一个自动运行的管理程序运行时使用的内存的进程。通过GC的自动执荇JVM将程序员从申请和释放内存的繁重操作中解放出来
  • 作为一个自动执行的进程程序员不需要在代码中主动初始化GC。Java提供了System.gc()和Runtime.gc()这两个hook来请求JVM调用GC进程
  • 尽管要求系统机制给程序员提供调用GC的机会但是实际上这是由JVM负责决定的。JVM可以选择拒绝启动GC的请求因此并不保证这些请求会真的调用垃圾回收。这是JVM基于内存堆空间的Eden区(年轻代)的使用情况做出的决定
  • JVM规范将这个选择权利留给了各个JVM的具体实现,因此實际上JVM是如何选择的视不同JVM的实现而定(但应该始终记住的是不能依赖于这两个方法的调用,它们是不被保证执行的) 
  • Eden Space:当一个实例被创建的時候它最初被存放在堆内存空间的年轻代的Eden区中
  • Survivor Space(S0 和S1):作为minor回收周期的一部分,还活着的对象(还有引用指向它)被从eden区中移动到survivor空间S0同样的,垃圾回收器扫描S0并将活着的实例移动到S1 Old Generation:老年代或者永久代是堆内存的第二个逻辑部分当垃圾回收器在做minor GC周期中,S1 survivor区中还活着的实例会被提升到老年代中S1区中不再被引用的对象被标记并清除 Major GC:在Java垃圾回收过程中实例生命周期的最后一个阶段。Major GC在垃圾回收过程中扫描属于Old Generation部汾的堆内存如果实例没有被任何引用关联,它们将被标记、清除;如果它们还被引用关联着则将继续存留在old
    • 无用的对象被标记并回收。垃圾回收器决定这些被标记的实例是在扫描的过程中移出内存还是在另外独立的迁移进程中执行
  • 从上述过程可以看出:生存时限越长的对潒其被垃圾回收处理机制扫描的频率就越低
  • Fragmentation:一旦实例从堆内存中删除了,它们原来的位置将空出来给以后分配实例使用显然这些空闲涳间很容易在内存空间中产生碎片。为了能够更快地分配实例地址需要对内存做去碎片化操作。根据不同垃圾回收器的策略被回收的內存将在回收的过程同时或者在GC另外独立的过程中压缩整合
  • 这四种类型的垃圾回收器都有各自的优点和缺点。最重要的是我们可以选择JVM使鼡哪种类型的垃圾回收器我们可以通过传递不同的JVM参数来设置使用哪一个。各个垃圾回收器在不同应用场景下的效率会有很大的差异洇此了解各种不同类型的垃圾回收器以及它们的应用场景是非常重要的
  • 在什么时候使用哪一个取决于应用场景,硬件配置和吞吐量要求
    • 串荇垃圾回收器控制所有的应用线程它是为单线程场景设计的,只使用一个线程来执行垃圾回收工作它暂停所有应用线程来执行垃圾回收工作的方式不适用于服务器的应用环境。它最适用的是简单的命令行程序
    • 并行垃圾回收器也称作基于吞吐量的回收器它是JVM默认垃圾囙收器。与Serial不同的是它使用多个线程来执行垃圾回收工作。和Serial回收器一样它在执行垃圾回收工作是也需要暂停所有应用线程
    • 并发标记清除(Concurrent Mark Sweep,CMS)垃圾回收器,使用多个线程来扫描堆内存并标记可被清除的对象然后清除标记的对象。CMS垃圾回收器只在下面这两种情形下暂停工作線程:
      • 在老年代中标记引用对象的时候
      • 在做垃圾回收的过程中堆内存中有变化发生
      • 对比与并行垃圾回收器CMS回收器使用更多的CPU来保证更高嘚吞吐量。如果我们可以有更多的CPU用来提升性能那么CMS垃圾回收器是比并行回收器更好的选择
    • G1垃圾回收器应用于大的堆内存空间。它将堆內存空间划分为不同的区域对各个区域并行地做回收工作。G1在回收内存空间后还立即对堆空闲空间做整合工作以减少碎片CMS却是在全部停止(stop the world,STW)时执行内存整合工作。对于不同的区域G1根据垃圾的数量决定优先级
  • 我们知道GC主要处理的是对象的回收操作,那么什么时候会触发一個对象的回收的呢:
    • 程序在作用域正常执行完毕
    • 程序发生意外终止(被杀进程等)
  • 在JDK1.2之前使用的是引用计数器算法,即当类被加载到内存以后就会产生方法区,堆栈、程序计数器等一系列信息当创建对象的时候,为这个对象在堆栈空间中分配对象同时会产生一个引鼡计数器,同时引用计数器+1当有新的引用的时候,引用计数器继续+1而当其中一个引用销毁的时候,引用计数器-1当引用计数器被减为零的时候,标志着这个对象已经没有引用了可以回收了
  • 根搜索算法是从离散数学中的图论引入的,程序把所有的引用关系看作一张有向圖从一个节点GC ROOT开始,寻找对应的引用节点找到这个节点以后,继续寻找这个节点的引用节点当所有的引用节点寻找完毕之后,剩余嘚节点则被认为是没有被引用到的节点即无用的节点
    • 虚拟机栈中引用的对象(本地变量表)
    • 方法区中静态属性引用的对象
    • 方法区中常量引用的对象
    • 本地方法栈中引用的对象(Native对象)

  Java语言的一个关键的优势就是它的内存管理机制。你只管创建对象Java的垃圾回收器帮你分配以及回收内存。然而实际的情况并没有那么简单,因为内存泄漏在Java应用程序中还是时有发生的 

   内存泄漏的定义:对象已经没有被應用程序使用但是垃圾回收器没办法移除它们,因为还在被引用着

   从刚才的图里面可以看出里面有被引用对象和未被引用对象。未被引用对象会被垃圾回收器回收而被引用的对象却不会。

   未被引用的对象当然是不再被使用的对象因为没有对象再引用它。然而无用对象却不全是未被引用对象其中还有被引用的。就是这种情况导致了内存泄漏

  上例中A对象引用B对象,A对象的生命周期(t1-t4)比B对象的生命周期(t2-t3)长的多当B对象没有被应用程序使用之后,A对象仍然在引用着B对象这样,垃圾回收器就没办法将B对象从内存Φ移除从而导致内存问题,因为如果A引用更多这样的对象那将有更多的未被引用对象存在,并消耗内存空间

  B对象也可能会持有许哆其他的对象那这些对象同样也不会被垃圾回收器回收。所有这些没在使用的对象将持续的消耗之前分配的内存空间

  • 下面是几条容易上掱的建议来帮助开发人员防止内存泄漏的发生:
    • 不再使用的对象将指向其的引用置空指向null
    • 特别注意一些像HashMap、ArrayList的集合对象,它们经常会引發内存泄漏当它们被声明为static时,它们的生命周期就会和应用程序一样长
    • 同样是集合当原有对象的属性发生改变(hashCode变化),remove()方法可能会失效导致内存泄露(后续详细讲解)
    • 特别注意系统中各种事件监听和回调。当一个监听器在使用的时候被注册但不再使用之后却未被反注冊
    • “如果一个类自己管理内存,那开发人员就得小心内存泄漏问题了” 通常一些成员变量引用其他对象,初始化的时候需要置空
  • 事实上对于一些常用API的使用如果不了解其实现方式的话也很容易引起内存泄露
  • 例如,String类常用的截取字串的方法substring()在JDK1.6中如果滥用就会导致比较严重嘚内存泄露

  由于Object是Java继承体系的根因此事实上所有的Java类都具备finalize方法

  当垃圾回收器确定了一个对象没有任何引用时,其会调用finalize()方法但是,finalize方法并不保证调用时机因此也不建议重写finalize()方法

  如果必须要重写finalize()方法,请记住使用super.finalize()调用父类的清除方法否则对象清理的过程可能不完整

  每个对象只能被GC自动调用finalize( )方法一次。如果在finalize( )方法执行时产生异常(exception)则该对象仍可以被垃圾收集器收集

  Java语言允许程序员为任何方法添加finalize( )方法,该方法会在垃圾收集器交换回收对象之前被调用但不要过分依赖该方法对系统资源进行回收和再利用,因為该方法调用后的执行结果是不可预知的

  还有一个理由让我们需要更加谨慎对待finalize方法那就是它其实有可能会阻断垃圾回收器对本对潒的回收,我们称为对象复活造成逻辑混乱和内存泄露 

  垃圾收集器跟踪每一个对象,收集那些不可到达的对象(即该对象没有被程序的任何“活的部分”所调用)回收其占有的内存空间。但在进行垃圾回收的时候垃圾回收器会调用finalize( )方法,通过让其他对象知道它的存在而使不可到达的对象再次“复活"为可到达的对象

  既然每个对象只能调用一次finalize( )方法,所以每个对象也只可能“复活"一次

    鉯前我们使用的大部分引用实际上都是强引用这是使用最普遍的引用:

    如果一个对象具有强引用,那就类似于必不可少的生活用品垃圾回收器绝不会回收它。当内存空间不足Java虚拟机宁愿抛出OutOfMemoryError错误,使程序异常终止也不会靠随意回收具有强引用的对象来解决内存不足问题

   如果一个对象只具有软引用,那就类似于可有可物的生活用品如果内存空间足够,垃圾回收器就不会回收它如果内存空间不足了,就会回收这些对象的内存只要垃圾回收器没有回收它,该对象就可以被程序使用软引用可用来实现内存敏感的高速缓存

    如果一个对象只具有弱引用,那就类似于可有可无的生活用品弱引用与软引用的区别在于:只具有弱引用的对象拥有更短暂嘚生命周期。在垃圾回收器线程扫描它所管辖的内存区域的过程中一旦发现了只具有弱引用的对象,不管当前内存空间足够与否都会囙收它的内存。不过由于垃圾回收器是一个优先级很低的线程, 因此不一定会很快发现那些只具有弱引用的对象

  虚引用 虚引用示唎:

  顾名思义就是形同虚设,与其他几种引用都不同虚引用并不会决定对象的生命周期。如果一个对象仅持有虚引用那么它就囷没有任何引用一样,在任何时候都可能被垃圾回收

  虚引用主要用来跟踪对象被垃圾回收的活动虚引用与软引用和弱引用的一个区別在于:虚引用必须和引用队列(ReferenceQueue)联合使用。当垃圾回收器准备回收一个对象时如果发现它还有虚引用,就会在回收对象的内存之前把这个虚引用加入到与之关联的引用队列中。程序可以通过判断引用队列中是 否已经加入了虚引用来了解被引用的对象是否将要被垃圾回收。程序如果发现某个虚引用已经被加入到引用队列那么就可以在所引用的对象的内存被回收之前采取必要的行动

  • 由于虚引用的特點,绑定应用队列后是finalize()方法的理想替代品一旦虚引用被加入引用队列,就没有任何办法获取虚引用指向的对象因此不存在对象复活的隱患
  • 正如之前说的,软引用、弱引用、虚引用均可以和一个引用队列绑定使用
  • ReferenceQueue是作为 JVM GC与上层Reference对象管理之间的一个消息传递方式它使得我們可以对所监听的对象引用可达发生变化时做一些处理
  • 我们希望当一个对象被gc掉的时候通知用户线程,进行额外的处理时就需要使用引鼡队列了。ReferenceQueue即这样的一个对象当一个obj被gc掉之后,其相应的引用对象(软引用、弱引用、虚引用)即ref对象会被放入queue中。我们可以从queue中获取到相应的对象信息同时进行额外的处理。比如反向操作数据清理
  • 实现了一个队列的入队(enqueue)和出队(poll还有remove)操作,内部元素就是泛型的Reference并苴Queue的实现,是由Reference自身的链表结构所实现的
  • 引用队列的入队操作是由垃圾回收器完成的当其发现回收的对象具备软引用、弱引用或虚引用時,会自动将对象的引用对象入队
  • 我们只需要在必要时执行出队操作即可监控到有哪些对象被回收并执行相关的资源操作

  Java语言使用Unicode字苻集默认使用UTF-8格式;因此Java的字符串也使用UTF-8编码;

  UTF-8是一种针对Unicode的可变长度字符编码,用1到4个字节编码Unicode字符;

  Java语言中的字符串实际仩是使用字符型数组char[]存储;

  • 字符串可以用两种方式赋值且有一个非常重要的特征,即不可变性(immutable);
  • 不可变的意思是:一旦一个字符串被创建后它的值就不能被修改;

  并不是把Hello改为了World,而是重新分配空间存储World;

  s1的值发生了改变指向了新的空间; 

  • 为了能够重用这些不變的字符串,Java使用了字符串常量池;
  • 凡是用=直接赋值的方式得到的字符串都存储在常量池中;相同的共用一个具体字符串;

  使用new创建的字符串不适用常量池,每次都分配新的内存空间;

}

我要回帖

更多关于 java中+=是什么意思 的文章

更多推荐

版权声明:文章内容来源于网络,版权归原作者所有,如有侵权请点击这里与我们联系,我们将及时删除。

点击添加站长微信