简述Java回收机制 - 初学者必看

news/2024/5/18 12:57:35 标签: java, jvm, c++, 算法, delete, bbs
一.谁在做Garbage Collection?

       一种流行的说法:在C++里,是系统在做垃圾回收;而在Java里,是Java自身在做。

在C++里,释放内存是手动处理的,要用delete运算符来释放分配的内存。这是流行的说法。确切地说,是应用认为不需要某实体时,就需用delete告诉系统,可以回收这块空间了。这个要求,对编码者来说,是件很麻烦、很难做到的事。随便上哪个BBS,在C/C++版块里总是有一大堆关于内存泄漏的话题。

Java采用一种不同的,很方便的方法:Garbage Collection。垃圾回收机制放在JVM里。JVM完全负责垃圾回收事宜,应用只在需要时申请空间,而在抛弃对象时不必关心空间回收问题。

二.对象在啥时被丢弃?

       在C++里,当对象离开其作用域时,该对象即被应用抛弃。

是对象的生命期不再与其作用域有关,而仅仅与引用有关。

       Java的垃圾回收机制一般包含近十种算法。对这些算法中的多数,我们不必予以关心。只有其中最简单的一个:引用计数法,与编码有关。

       一个对象,可以有一个或多个引用变量指向它。当一个对象不再有任何一个引用变量指向它时,这个对象就被应用抛弃了。或者说,这个对象可以被垃圾回收机制回收了。

这就是说,当不存在对某对象的任何引用时,就意味着,应用告诉JVM:我不要这个对象,你可以回收了。

JVM的垃圾回收机制对堆空间做实时检测。当发现某对象的引用计数为0时,就将该对象列入待回收列表中。但是,并不是马上予以销毁。

三.丢弃就被回收?

该对象被认定为没有存在的必要了,那么它所占用的内存就可以被释放。被回收的内存可以用于后续的再分配。

但是,并不是对象被抛弃后当即被回收的。JVM进程做空间回收有较大的系统开销。如果每当某应用进程丢弃一个对象,就立即回收它的空间,势必会使整个系统的运转效率非常低下。

前面说过,JVM的垃圾回收机制有多个算法。除了引用计数法是用来判断对象是否已被抛弃外,其它算法是用来确定何时及如何做回收。JVM的垃圾回收机制要在时间和空间之间做个平衡。

因此,为了提高系统效率,垃圾回收器通常只在满足两个条件时才运行:即有对象要回收且系统需要回收。切记垃圾回收要占用时间,因此,Java运行时系统只在需要的时候才使用它。因此你无法知道垃圾回收发生的精确时间。

四.没有引用变量指向的对象有用吗?

       前面说了,没挂上引用变量的对象是被应用丢弃的,这意味着,它在堆空间里是个垃圾,随时可能被JVM回收。

不过,这里有个不是例外的例外。对于一次性使用的对象(有些书称之为临时对象),可以不用引用变量指向它。举个最简单也最常见的例子:

System.out.println(“I am Java!”);

就是创建了一个字符串对象后,直接传递给println()方法。

五.应用能干预垃圾回收吗?

       许多人对Java的垃圾回收不放心,希望在应用代码里控制JVM的垃圾回收运作。这是不可能的事。对垃圾回收机制来说,应用只有两个途径发消息给JVM。第一个前面已经说了,就是将指向某对象的所有引用变量全部移走。这就相当于向JVM发了一个消息:这个对象不要了。第二个是调用库方法System.gc(),多数书里说调用它让Java做垃圾回收。

       第一个是一个告知,而调用System.gc()也仅仅是一个请求。JVM接受这个消息后,并不是立即做垃圾回收,而只是对几个垃圾回收算法做了加权,使垃圾回收操作容易发生,或提早发生,或回收较多而已。

       希望JVM及时回收垃圾,是一种需求。其实,还有相反的一种需要:在某段时间内最好不要回收垃圾。要求运行速度最快的实时系统,特别是嵌入式系统,往往希望如此。

       Java的垃圾回收机制是为所有Java应用进程服务的,而不是为某个特定的进程服务的。因此,任何一个进程都不能命令垃圾回收机制做什么、怎么做或做多少。

六.对象被回收时要做的事

一个对象在运行时,可能会有一些东西与其关连。因此,当对象即将被销毁时,有时需要做一些善后工作。可以把这些操作写在finalize()方法(常称之为终止器)里。

protected void finalize()

{

  // finalization code here

}

这个终止器的用途类似于C++里的析构函数,而且都是自动调用的。但是,两者的调用时机不一样,使两者的表现行为有重大区别。C++的析构函数总是当对象离开作用域时被调用。这就是说,C++析构函数的调用时机是确定的,且是可被应用判知的。但是,Java终止器却是在对象被销毁时。由上所知,被丢弃的对象何时被销毁,应用是无法获知的。而且,对于大多数场合,被丢弃对象在应用终止后仍未销毁。

在编码时,考虑到这一点。譬如,某对象在运作时打开了某个文件,在对象被丢弃时不关闭它,而是把文件关闭语句写在终止器里。这样做对文件操作会造成问题。如果文件是独占打开的,则其它对象将无法访问这个文件。如果文件是共享打开的,则另一访问该文件的对象直至应用终结仍不能读到被丢弃对象写入该文件的新内容。

至少对于文件操作,编码者应认清Java终止器与C++析构函数之间的差异。

那么,当应用终止,会不会执行应用中的所有finalize()呢?据Bruce Eckel在Thinking in Java里的观点:“到程序结束的时候,并非所有收尾模块都会得到调用”。这还仅仅是指应用正常终止的场合,非正常终止呢?
因此,哪些收尾操作可以放在finalize()里,是需要酌酎的 

http://www.niftyadmin.cn/n/610566.html

相关文章

【JokerのZYNQ7020】AXI_UARTLITE_LINUX。

软件环境:vivado 2017.4 硬件平台:XC7Z020 工程还是之前那篇的那个工程,今天说下怎么把axi_uartlite加到linux下,并使用起来。 工程编译完成,生成设备树以后,看眼设备树pl.dtsi,确定是如…

C语言数据结构-第六章、树和二叉树单元测试-电大同步进度

第六章 单元测验1 1单选(5分) 已知一算术表达式的中缀形式为 A-B/CD*E,前缀形式为-A/BC*DE,其后缀形式为( )。 A.ABC/-DE* B. A-BC/DE* C.AB/C-D*E D. ABCDE/-* 2单选(5分) 有关二叉树下列说法正确的是( )。 A.二叉…

【JokerのZYNQ7020】UDP_ECHO_LINUX。

软件环境:vivado 2017.4 硬件平台:XC7Z020 工程与前一篇linux下操作axi_uartlite用的是同一个工程,所以网口自然用的是PS端的网口,主要是想要说明zynq平台linux下怎样通过网口收发udp的数据。 由于用的是PS端的网口&#x…

通过代码快速上手C语言数据结构---树

6.1-6.3遍历二叉树 6.1先序遍历二叉树 6.2中序遍历二叉树 6.3后序遍历二叉树 #include <stdio.h> #include <malloc.h> #include <conio.h>typedef char DataType;typedef struct Node {DataType data;struct Node* LChild;struct Node* RChild; }BiTNod…

【JokerのZYNQ7020】TCP_ECHO_LINUX。

软件环境&#xff1a;vivado 2017.4 硬件平台&#xff1a;XC7Z020 老样子&#xff0c;工程与上一篇UDP的和Uartlite的工程还是同一个工程&#xff0c;上篇说了udp数据收发&#xff0c;这篇自然而然的继续说tcp数据收发&#xff0c;共分两个部分&#xff0c;分别是tcp_se…

Java类的完整构造执行顺序

这里只说一个完整的结果&#xff0c;至于为什么是这样的顺序&#xff0c;可以参考我以前的文章&#xff1a;深入剖析java类的构造方式 如果父类有静态成员赋值或者静态初始化块&#xff0c;执行静态成员赋值和静态初始化块 如果类有静态成员赋值或者静态初始化块&#xff0c;执…

通过代码快速上手C语言数据结构-栈与队列

3.1-3.4顺序栈 3.1初始化顺序栈 3.2顺序栈进栈运算 3.3顺序栈出栈运算 3.4顺序栈取栈顶元素运算 顺序栈头文件 #define TRUE 1 #define FALSE 0 #define Stack_Size 50/*顺序栈*/typedef struct {StackElementType elem[Stack_Size]; /*用来存放栈中元素的一维数组*/int …

【JokerのZYNQ7020】USER_TYPE_ETH_LINUX

软件环境&#xff1a;vivado 2017.4 硬件平台&#xff1a;XC7Z020 之前已经说了在linux下&#xff0c;怎么建立UDP、TCP-server、TCP-client的应用&#xff0c;而这些都是带有特定协议的以太网数据包&#xff0c;所以就好奇啦&#xff0c;有没有办法发送自定义type类型的…