【转】.net框架读书笔记---CLR内存管理\垃圾收集(六)

对象代龄

  代龄是旨在提高垃圾收集器性能的一种机制。有以下几点:

  1. 对象越新,其生存期越短;
  2. 对象越老,其生存期越长;
  3. 对托管堆的一部分执行垃圾收集要比对整个托管堆执行垃圾收集速度要快。

  在托管堆初始化时,其中不包含任何对象。这时添加到托管堆中的对象被称为第0代对象。第0代对象就是新构造的对象,垃圾收集器还没有对他们执行过任何的检查,加入新启动一个应用程序托管堆,分配了5个对象(A到E),经过一段时间后,对象C和E将变为不可达对象。

  当CLR初始化时,它会为第0代对象选择一个阈值,假定为256KB。(实际可能会与此不同)当分配新对象导致第0代对象超过了为其设定的阙值时,垃圾收集器就必须启动了。假定上面的A到E总共占用了256KB,那么当F开始被分配时,垃圾收集器就会启。垃圾收集器判定对象C和E为垃圾,因此会压缩对象D使其邻接于对象B。此次垃圾收集器中存活下来的对象(对象A,B,D)将被认为是第1代对象。经过一轮垃圾收集的检查加强的存活下来并且光荣的升官加爵。

  在第一次垃圾收集执行后,0代对象暂时空缺。但很快,随着应用程序的运行,又有新的对象被分配而成为第0代对象。新加入的F到K就变为了0代对象变为了准炮灰。随着应用程序的运行,对象B(一代长老),H(0)和J(0)又成为不可达对象,它们的内存也要在某一点被回收。

  现在假设应用程序又试图分配对象L,这将在一次使第0代对象超过它的阙值容量,于是又必须开始执行第二次垃圾收集。在这前面,垃圾收集器必须要判定要收集那些代的对象。垃圾收集器期会为0代对象选择一个阙值(大约256KB),其实也会为第1代对象选择一个阙值容量。假设第1代对象的阙值容量为2MB。

  当第二次垃圾收开始执行时,它也会查看第1代对象占用了多少内存,在本例中,由于第一代对象(A,B,D)占用内存远少于2MB,所以垃圾收集器只会去检查第0代对象。由于新创建的对象的生存期比较短,所以第0代对象成为垃圾对象的数量会比较多,因此对第0代对象执行垃圾收集将有可能比较多的内存,忽略掉第1代对象提高垃圾收集的速度。

  忽略第1代对象的意义不仅仅在于不对它们执行垃圾收集。实际上,更重要的是垃圾收集器不用再遍历整个托管堆中的对象了,可以提高垃圾收集器的性能。说忽略代龄较大的对象的一些内部引用,而不是全部内部引用是因为代龄较大的对象有可能引用一个新的对象。为了确保这些代龄较大的对象中的一些新对象也能被检测到,垃圾收集器使用JIT编译器内部的一种机制来在对象的内部引用字段改变时设置一个相应的为标记。这使得垃圾收集器可以知道自从上次垃圾收集执行后以来,那些代龄较大的对象已经被应用程序所改写。这样垃圾收集器只需检查它们是否引用了第0代对象就可以了。

  生存期比较长的对象将趋向于继续存活,所以第一代对象继续成为可达对象的可能性比较大。所以第一代对象继续成为可达对象的可能性比较大。因此垃圾收集器检查第一代对象,很有可能找不到很多垃圾,能够回收的内存也就有限,这样收集第一代很有可能浪费大量的时间。如果垃圾对象位于第一代,那么它仍然健在。第二次垃圾收集后。托管堆中存在

  A--B---D---F--G---I---K(均为一代长老了)

  第0代对象中的幸存者有升官加爵了变为1代,因为垃圾收集器没有检查第一代,B内存没有被收集。虽然已经成为不可达对象。但是由于没有经历洗礼,所以没有升官加爵;

  此时0代为空,新的对象继续加入成为了0代长老(L到O),过一段时间,G、L、M又成为不可达对象。此时托管堆如下:

  一代A---B---D---F---G---I---K 0代L---M---N---O

  此时1代中堆着N多的不可达垃圾包括B。

  此时对象P的加入又是0代超标了,继续垃圾收集。因为1代的容量2MB,没有超标,继续不对1代收集(B,G为1代中的蛀虫);本次垃圾收集完毕后,托管堆如下:

  一代A---B---D---F---G---I---K---N----O

  0代被适时杀死,一代里面的蛀虫继续增长,没有控制是不行的,结果一代的容量达到了其阙值2MB,此时应用程序分配P到S对象,这时0代又满了。此时托管堆的分配情况:

  一代A---B---D---F---G---I---K---N----O  0代P----Q----R----S

  此时应用程序试图分配对象T,当然是没有地方了,0,1代都满了,需要清理了,这时发现一代对象的内存容量已经达到了2MB,然而一代中存在着大量幸存下来的垃圾。本次垃圾收集器是公平的,同时收集第0代和第1代中所有的垃圾对象,痛苦,所以的垃圾一扫而光了,此时托管堆为:

  二代D---F---I----N----O  二代Q-----S(0代空了)

  和前面一样,本次垃圾收集完毕后,幸存下来的将被提升为1代,而一代中存活下来的被荣升为2代了,0代为空了,在产生第二代对象之前,系统有可能已经执行了多次垃圾收集,但只有在第一代对象的内存总量达到其阙值时,垃圾收集器才会检查一代对象,而此时系统可能已经对0代对象执行了N多次垃圾收集了。

  CLR托管堆只支持3个代龄,0,1,2,CLR初始化时都会为其设置阙值容量(大约为,256KB,2MB,10MB),

  CLR垃圾收集器是一个自调结的玩意,他会根据对象的状态,变为不可达的频率自适应的调节阙值。实在是高啊。

 

  CLR的垃圾收集原理其实也给人一人生道理,要想活得更长,就需向更高一层走,而要想往更高一层走就要经过残酷的洗礼。剩者为王。

 

  • 屠刀总是先指向弱者,因为强者数大根深啊,杀一次得不偿失
  • 强者拥有更多的资源和更多的容忍
  • 再强也不要超过别人给的限度,否则权利是人家给的,人家不高兴了随时可以拿回去

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/436784.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MySQL数据库开发理念_mysql之数据库基本理念

数据储存的问题:数据冗余和不一致数据访问困哪数据孤立完整性原子性问题例如AB2个账户,从A中向B转钱,不管如何转,A和B的总量是保持不变的。只要A减少,B就增加。并发访问异常安全性问题文件的分层:表示层&am…

【转】.net框架读书笔记---CLR内存管理\垃圾收集(七)

编程控制垃圾收集器 System.GC类型为应用程序提供了直接控制垃圾收集器的一些方法,可以通过GC.MaxGeneration来查询托管堆支持的最大代龄,目前为2。 通过下面方法执行垃圾收集器 GC.Collect(int);传递代龄,传递0,收集0代&#xff…

少年自学python笔记_自学python 笔记

print() 输入input() 输出python能够处理的数据类型:整数、浮点数、字符串("\"转义符,\n 换行)、布尔值/布尔代数(True、False(and【或】、or【于】、not【非】))(and:只有所有都为True,and运算结果才为True)(or:只要一个为True,or…

【转】深入浅出图解C#堆与栈 C# Heap(ing) VS Stack(ing) 第四节 参数传递对堆栈的影响 1

前言 虽然在.Net Framework 中我们不必考虑内在管理和垃圾回收(GC),但是为了优化应用程序性能我们始终需要了解内存管理和垃圾回收(GC)。另外,了解内存管理可以帮助我们理解在每一个程序中定义的每一个变量是怎样工作的。 简介 这篇文章我们将介绍一些方…

java解析string_java读取文件内容为string字符串的方法

直接就把项目中的方法贴出来吧/*** 读出城市列表文件*/private String readCityFile() {File file02 new File(path_xinfu, "/cityList.json");FileInputStream is null;StringBuilder stringBuilder null;try {if (file02.length() ! 0) {/*** 文件有内容才去读文…

【转】深入浅出图解C#堆与栈 C# Heap(ing) VS Stack(ing) 第四节 参数传递对堆栈的影响 2

前言 虽然在.Net Framework 中我们不必考虑内在管理和垃圾回收(GC),但是为了优化应用程序性能我们始终需要了解内存管理和垃圾回收(GC)。另外,了解内存管理可以帮助我们理解在每一个程序中定义的每一个变量是怎样工作的。 简介 继续上篇未完成的“参数传…

java 引用被回收_java GC 静态List 如果没有引用会被回收吗

垃圾收算法1.引用计数法(Reference Counting Collector)2.tracing算法(Tracing Collector)3.compacting算法(Compacting Collector)4.copying算法(Coping Collector)5。generation算法(Generational Collector)6.adaptive算法(Adaptive Collector)一个新的对象被创建&#xff0…

【转】深入浅出图解C#堆与栈 C# Heap(ing) VS Stack(ing) 第五节 引用类型复制问题及用克隆接口ICloneable修复

前言 虽然在.Net Framework 中我们不必考虑内在管理和垃圾回收(GC),但是为了优化应用程序性能我们始终需要了解内存管理和垃圾回收(GC)。另外,了解内存管理可以帮助我们理解在每一个程序中定义的每一个变量是怎样工作的。 简介 这一节我们将介绍引用类型…

linux virt java_Linux下Java环境安装

本节主要讲解Linux(Centos 6.5)下Java环境的安装1. 卸载机器上默认安装的JDK在Linux环境下一般会默认安装jdk,为了自己项目的开发部署,一般情况要重新装jdk,而且自己装的Jdk相对来说易控制版本,稳定性更高。所以以下是我卸载预装J…

【转】深入浅出图解C#堆与栈 C# Heap(ing) VS Stack(ing) 第六节 理解垃圾回收GC,提搞程序性能****

前言 虽然在.Net Framework 中我们不必考虑内在管理和垃圾回收(GC),但是为了优化应用程序性能我们始终需要了解内存管理和垃圾回收(GC)。另外,了解内存管理可以帮助我们理解在每一个程序中定义的每一个变量是怎样工作的。 简介 这一节我们将介绍垃圾回…

java下拉列表 动态_【示例】教你简单用Java写一个动态更新的下拉列表(无数据库)...

动态更新下拉列表varxmlHttp;functioncreatXMLHttpRequest(){if(window.ActiveXObject){xmlHttpnewActiveXObject("Microsoft.XMLHTTP");}else if(window.XMLHttpRequest){xmlHttpnewXMLHttpRequest();}}functionupdateSelect(){varselecteddocument.all.slt1.value;…

【转】分布式事务的常见解决方案

一、事务起步 1. 什么是事务 事务这种东西大家都耳熟能详了,通常指由一组操作组成的一个工作单元,这一整个组合要么全部成功,要么全部失败。 2. 本地事务 在计算机系统中,更多的是通过关系型数据库来控制事务,这是…

深入解析java web_java进阶--深入分析java Web

第一章: 深入了解Web请求整理本书的内容与之前的采用相同的方式,主要目的还是为了可以仔细的阅读。整理自己的见解。这本书整体的感觉很好,思路很清晰,最近就发现,国人写的文字和外国译文相差很大,主要体现…

【转】修饰符new将父类中的该方法隐藏掉有什么意义 不隐藏有什么弊端

这是一个C#语法的问题。子类如果要重写父类的方法的话,virtual-override一定成对的。 子类不重写,而是创建一个属于自己的同名方法,就最好加个new。如果不加new也等于new,但是编译器都会提醒你,加个标识比较好。 区别见…

java s.charat_Java中s.charAt(index)用于提取字符串s中的特定字符操作

charAt(int index)方法是一个能够用来检索特定索引下的字符的String实例的方法.charAt()方法返回指定索引位置的char值。索引范围为0~length()-1.如: str.charAt(0)检索str中的第一个字符,str.charAt(str.length()-1)检索最后一个字符.警告:在字符串s中越界访问字符…

【转】.NET框架简介

.NET 框架是由微软开发的软件开发平台,其最主要的两个组成部分是公共语言运行时 (CLR) 和框架类库 (FCL),基础类库 (BCL)是框架类库的一个子集。 .NET 框架简介 下图展示了 .NET 框架的主要结构。 其中,最下层的无疑就是操作系统了。 在 …

eclipse java maven_java – 非常轻量级的Eclipse-Maven集成 – 仅...

我找到了一个非常适合我所描述的需求/用例的解决方案:1.我在Eclipse项目根目录中创建了非常小的pom文件:4.0.0com.sobczyk.piotrmvn-eclipse-test1.0.0srcbinlog4jlog4j1.2.16这个.pom文件由三部分组成:>必需的Maven东西,即. modelVersion…

【转】Path.Combine (合并两个路径字符串)方法的一些使用细节

System.IO.Path.Combine 简单来说,就是合并两个路径字符串。 比如下面这个调用, Path.Combine(“C:\11”,“aa.txt”) 返回的字符串路径为: C:\11\aa.txt 这个方法的声明如下: public static string Combine ( string path1, st…

java 正则表达式 开头 结尾_Java-正则表达式匹配 #开头结尾

引包import java.util.regex.Matcher;import java.util.regex.Pattern;方法1:// 匹配 #开头结尾中,#以及中间得字符串 #xxx 替换为 "" #123 匹配#123String tableModle "#123#2#3#4";Pattern pPattern.compile("…

java比赛题目_【蓝桥杯2016第七届比赛题目】JAVA A组

1 煤球数目有一堆煤球,堆成三角棱锥形。具体:第一层放1个,第二层3个(排列成三角形),第三层6个(排列成三角形),第四层10个(排列成三角形),....如果一共有100层,共有多少个煤球?请填表…