什么是垃圾回收(转)

news/2024/7/20 12:39:50 标签: c/c++, 内存管理, php

 

本文摘自我们几周后即将出版的Garbage Collection Handbook一书的样章。同时也让你能熟悉下垃圾回收的基础知识——这选自该书的第一章。 

乍一看,垃圾回收所做的事情应当恰如其名——查找并清除垃圾。事实上却恰恰相反。垃圾回收会跟踪所有仍在使用的对象,然后将剩余的对象标记为垃圾。牢记了这点之后,我们再来深入地了解下这个被称为“垃圾回收”的自动化内存回收在JVM中到底是如何实现的。 

手动管理内存 

在介绍现代版的垃圾回收之前,我们先来简单地回顾下需要手动地显式分配及释放内存的那些日子。如果你忘了去释放内存,那么这块内存就无法重用了。这块内存被占有了却没被使用。这种场景被称之为内存泄露。 

下面是用C写的一个手动管理内存的简单例子: 

Java代码  
  1. int send_request() {  
  2.     size_t n = read_size();  
  3.     int *elements = malloc(n * sizeof(int));  
  4.    
  5.     if(read_elements(n, elements) < n) {  
  6.         // elements not freed!  
  7.         return -1;  
  8.     }  
  9.    
  10.     // …  
  11.    
  12.     free(elements)  
  13.     return 0;  
  14. }  


可以看到,你很容易就会忘了释放内存。内存泄露曾经是个非常普遍的问题。你只能通过不断地修复自己的代码来与它们进行抗争。因此,需要有一种更优雅的方式来自动释放无用内存,以便减少人为错误的可能性。这种自动化过程又被称为垃圾回收(简称GC)。 

智能指针 
自动垃圾回收早期的一种实现便是引用计数。你知晓每一个对象被引用了几次,当计数器归0的时候,这个对象就可以被安全地回收掉了。C++的共享指针就是一个非常著名的例子: 

Java代码  
  1. int send_request() {  
  2.     size_t n = read_size();  
  3.     stared_ptr<vector<int>> elements   
  4.               = make_shared<vector<int>&gt();  
  5.    
  6.     if(read_elements(n, elements) < n) {  
  7.         return -1;  
  8.     }  
  9.    
  10.     return 0;  
  11. }  


我们使用的sharedptr会记录这个对象被引用的次数。如果你将它传递给别人则计数加一,当它离开了作用域后便会减一。一旦这个计数为0,sharedptr会自动地删除底层对应的vector。当然这只是个示例,因为也有读者指出来了,这个在现实中是不太可能出现的,但作为演示是足够了。 

自动内存管理 

在上面的C++代码中,我们还得显式地声明我们需要使用内存管理。那如果所有的对象都采用这个机制会怎样呢?那简直就太方便了,这样开发人员便无需考虑清理内存的事情了。运行时会自动知晓哪些内存不再使用了,然后释放掉它。也就是说,它自动地回收了这些垃圾。第一代的垃圾回收器是1959年Lisp引入的,这项技术迄今为止一直在不断演进。 

引用计数 
刚才我们用C++的共享指针所演示的想法可以应用到所有的对象上来。许多语言比如说Perl, Python以及PHP,采用的都是这种方式。这个通过一张图可以很容易说明: 

 


绿色的云代表的是程序中仍在使用的对象。从技术层面上来说,这有点像是正在执行的某个方法里面的局部变量,亦或是静态变量之类的。不同编程语言的情况可能会不一样,因此这并不是我们关注的重点。 

蓝色的圆圈代表的是内存中的对象,可以看到有多少对象引用了它们。灰色圆圈的对象是已经没有任何人引用的了。因此,它们属于垃圾对象,可以被垃圾回收器清理掉。 

看起来还不错对吧?没错,不过这里存在着一个重大的缺陷。很容易会出现一些孤立的环,它们中的对象都不在任何域内,但彼此却互相引用导致引用数不为0。下面便是一个例子: 

 


看到了吧,红色部分其实就是应用程序不再使用的垃圾对象。由于引用计数的缺陷,因此会存在内存泄露。 

有几种方法可以解决这一问题,比如说使用特殊的“弱”引用,或者使用一个特殊的算法回收循环引用。之前提到的Perl,Python以及PHP等语言,都是使用类似的方法来回收循环引用的,不过这已经超出本文讲述的范围了。我们准备详细介绍下JVM所采用的方法。 

标记删除 

首先,JVM对于对象可达性的定义要明确一些。它可不像前面那样用绿色的云便含糊了事的,而是有着非常明确及具体的垃圾回收根对象(Garbage Collection Roots)的定义: 

  • 局部变量
  • 活动线程
  • 静态字段
  • JNI引用
  • 其它(后面将会讨论到)


JVM通过标记删除的算法来记录所有可达(存活)对象,同时确保不可达对象的那些内存能够被重用。这包含两个步骤: 

  • 标记是指遍历所有可达对象,然后在本地内存中记录这些对象的信息
  • 删除会确保不可达对象的内存地址可以在下一次内存分配中使用。


JVM中的不同GC算法,比如说Parallel Scavenge,Parallel Mark+Copy, CMS都是这一算法的不同实现,只是各阶段略有不同而已,从概念上来讲仍然是对应着上面所说的那两个步骤。 

这种实现最重要的就是不会再出现泄露的对象环了: 

 


缺点就是应用程序的线程需要被暂停才能完成回收,如果引用一直在变的话你是无法进行计数的。这个应用程序被暂停以便JVM可以收拾家务的情况又被称为Stop The World pause(STW)。这种暂停被触发的可能性有很多,不过垃圾回收应该是最常见的一种。 

http://www.iteye.com/news/30630

 


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

相关文章

33九九乘法表

33 九九乘法表 作者: Turbo时间限制: 1S章节: 循环 问题描述 : 按如下形式输出九九乘法表&#xff1a; 1447766779331024994.jpg 输入说明 : 输入一个整数n&#xff08;1<n<9&#xff09;&#xff0c;表示输出n*n乘法表。 输出说明 : 输出三个n*n乘法表&#xff…

.NET平台下的微信SDK(Rabbit.WeiXin)开源发布

在上一篇文章《RabbitHub开源情况及计划》上有提及到了一个新的开源项目——微信SDK&#xff0c;经过几天的努力现在开源发布Beta1版本。 目录 前言特点功能支持的消息类型请求消息事件消息响应消息消息处理中间件支持的API暂不支持的API关于性能关于易扩展性关于易使用性关于架…

34放大的X

34 放大的X 作者: xxx时间限制: 1S章节: 循环 问题描述 : 请你编程画一个放大的’X’。 如2*2的’X’应如下所示&#xff1a; XX XX 5*5的’X’如下所示&#xff1a; X X X X X X X X X 输入说明 : 输入数据第一行是一个整数T&#xff0c;表示有T组测试数据&…

简要介绍如何集成Vitamio安卓版SDK

1.下载VitamioBundle的最新稳定&#xff0c;这里下载的是最新版4.2.2。 2.解压缩后&#xff0c;导入 Vitamio 库工程&#xff08;即vitamio&#xff09;和Demo工程&#xff08;即vitamio--sample&#xff09;到 Eclipse&#xff0c;详细步骤如下图。 步骤&#xff1a;通过File …

35空心三角形

35 空心三角形 作者: xxx时间限制: 1S章节: 循环 问题描述 : 把一个字符三角形掏空&#xff0c;就能节省材料成本&#xff0c;减轻重量&#xff0c;但关键是为了追求另一种视觉效果。在设计的过程中&#xff0c;需要给出各种花纹的材料和大小尺寸的三角形样板&#xff0c;通…

给Ubuntu软件升级命令

以非root用户更新系统 sudo: sudo是linux系统管理指令&#xff0c;是允许系统管理员让普通用户执行一些或者全部的root命令的一个工具&#xff0c;如halt&#xff0c;reboot&#xff0c;su等等。这样不仅减少了root用户的登陆 和管理时间&#xff0c;同样也提高了安全性。Sudo不…

37求奇数的乘积

37 求奇数的乘积 作者: xxx时间限制: 1S章节: 循环 问题描述 : 给你n个整数&#xff0c;求他们中所有奇数的乘积。 输入说明 : 输入数据包含两行&#xff0c;第一行为一个数为n&#xff0c;表示第二行将输入n个整数。你可以假设这n个数据中必定至少存在一个奇数。 输出说…

一台windows机器上同时部署多个tomcat服务

2019独角兽企业重金招聘Python工程师标准>>> 如果现在一台机器上已经部署了一个tomcat服务&#xff0c;无论这个tomcat是否已经注册为服务了&#xff0c;或者没有注册windows服务&#xff0c;或者注册了&#xff0c;都没关系。都可以采用下面的方法实现。 如果该tom…