内存是稀缺的资源哪怕内存一块钱一条!如果在编程中使用不当再大的内存也会耗光
一认识Java的自动垃圾回收
垃圾回收是Java语言的一大特性方便了编程是以消耗性能为代价的而垃圾在这里只无用的对象而C++是需要程序员自己写析构函数来释放内存的麻烦也有可能忘记而导致内存洩露
Java语言对内存的分配管理是通过JVM内部机制决定的程序员可以不关心其处理
二垃圾回收的原理和意义
Java虚拟机中有个称之为垃圾回收器的东西实际上这个东西也许真正不存在或者是已经集成到JVM中了但这无关紧要我们仍然可以称为为垃圾回收器
垃圾回收器的作用是查找和回收(清理)无用的对象以便让JVM更有效的使用内存
垃圾回收器的运行时间是不确定的由JVM决定在运行时是间歇执行的虽然可以通过Systemgc()来强制回收垃圾但是这个命令下达后无法保证JVM会立即响应执行但经验表明下达命令后会在短期内执行你的请求JVM通常会感到内存紧缺时候去执行垃圾回收操作
垃圾回收过于频繁会导致性能下降过于稀疏会导致内存紧缺这个JVM会将其控制到最好不用程序员担心但有些程序在短期会吃掉大量内存而这些恐怖的对象很快使用结束了这时候也许有必要强制下达一条垃圾回命令这是很有必要的以便有更多可用的物理内存
从上面了解到没有用的对象就是垃圾准确的说当没有任何线程访问一个对象时该对象就符合垃圾回收的条件
对于String存在一个字符串池这个不属于本文讨论的范围字符串池中的垃圾回收算法和这里所讨论的垃圾回收完全是两码事但是不得不说的是字符串的胡乱拼接往往导致性能急剧下降尤其是在庞大的循环语句中拼接字符串就是在让程序慢性自杀这也是很多Java程序员容易犯的毛病
字符串既然是池就是为了缓沖为了有更高的命中率因此垃圾回收的频率也许会比JVM对象垃圾回收器要低很多
垃圾回收器仅仅能做的是尽可能保证可用内存的使用效率让可用内存得到高效的管理程序员可以影响垃圾回收的执行但不能控制
三通过编程影响垃圾回收
虽然程序员无法控制JVM的垃圾回收机制但是可以通过编程的手段来影响影响的方法是让对象符合垃圾回收条件
分别说来有一下几种
将无用对象赋值为null
重新为引用变量赋值比如
Person p = new Person(aaa);
p = new Person(bbb);
这样new Person(aaa)这个对象就是垃圾了——符合垃圾回收条件了
让相互联系的对象称为岛对象
Person p = new Person(aaa);
Person p = new Person(bbb);
Person p = new Person(ccc);
p=p; p=p; p=p;
p=null; p=null; p=null;
在没有对ppp置null之前它们之间是一种三角恋关系分别置null三角恋关系依然存在但是三个变量不在使用它们了三个Person对象就组成了一个孤岛最后死在堆上——被垃圾回收掉
强制的垃圾回收Systemgc()
实际上这里的强制是程序员的意愿建议什么时候执行是JVM的垃圾回收器说了算
调用垃圾回收也不一定能保证未使用的对象一定能从内存中删除
唯一能保证的是当你内存在极少的情况垃圾回收器在程序抛出OutofMemaryException之前运行一次
四很神秘的finalize()方法
finalize()方法的确很神秘是因为你不了解其原理
原理finalize()方法是Object中的方法
finalize()方法会在对象被垃圾回收之前被垃圾回收器调用一次这是Java语言的一种机制
finalize()方法在任何对象上最多只会被垃圾回收器调用一次
陷阱垃圾回收器无法保证垃圾对象能被回收因此finalize()方法也无法保证运行建议不要重写finalize()方法即使重写也不要在finalize()方法中写关键的代码
finalize()方法中可以把自己传递个别的对象这样就不是垃圾了避免了被回收但是当下次这个对象又符合垃圾回收的时候finalize()方法不会被调用第二次了而是直接被清理掉了
总结
理解了垃圾回收的前提是理解Java运行时的内存堆栈模型
理解Java垃圾回收的目的是为了对Java内存管理有个认识在编程时更有效的使用内存
不建议为了垃圾回收手动编写大量代码这是很愚蠢的做法可以通过简单的方式去影响即可
本文的讨论的垃圾回收排除String对象String的垃圾回收与String池有很很大关系目前还没有研究但是文中已经提及String使用中容易出现的问题