1.在java中,他的內存管理分為兩個方面.內存分配(創(chuàng)建java對象的時候)和內存回收,這兩個方面都是有JVM自動完成的.
java垃圾回收機制GC(Garbage Collection)
兩件事:1 內存回收 2.碎片管理
2需要注意3點:
(1)垃圾回收并不會按照程序員的要求,隨時驚喜GC
(2)垃圾回收機制并不會及時的清理內存,盡管有時候程序需要額外的內存
(3)程序員不能對垃圾回收進行控制
ps:
GC:能清除在堆上分配的內存(純java語言的所有對象都在堆上使用new分配內存)铃剔,而不能清除棧上分配的內存
finalize():對棧上的對象進行內存回收
java提供finalize()方法誉结,垃圾回收器準備釋放內存的時候忠怖,會先調用finalize()欲鹏。
(1).對象不一定會被回收。
(2).垃圾回收不是析構函數空扎。
(3).垃圾回收只與內存有關藏鹊。
(4).垃圾回收和finalize()都是靠不住的,只要JVM還沒有快到耗盡內存的地步转锈,它是不會浪費時間進行垃圾回收的盘寡。
3垃圾回收算法
1)串行回收(只用一個CPU)和并行回收(多個CPU才有用):串行回收是不管系統(tǒng)有多少個CPU,始終只用一個CPU來執(zhí)行垃圾回收操作撮慨,而并行回收就是把整個回收工作拆分成多個部分竿痰,每個部分由一個CPU負責,從而讓多個CPU并行回收砌溺。并行回收的執(zhí)行效率很高影涉,但復雜度增加,另外也有一些副作用规伐,如內存碎片增加蟹倾。
2)并發(fā)執(zhí)行和應用程序停止 :應用程序停止(Stop-the-world)顧名思義,其垃圾回收方式在執(zhí)行垃圾回收的同時會導致應用程序的暫停猖闪。并發(fā)執(zhí)行的垃圾回收雖然不會導致應用程序的暫停鲜棠,但由于并發(fā)執(zhí)行垃圾需要解決和應用程序的執(zhí)行沖突(應用程序可能在垃圾回收的過程修改對象),因此并發(fā)執(zhí)行垃圾回收的系統(tǒng)開銷比Stop-the-world高培慌,而且執(zhí)行時需要更多的堆內存豁陆。
3)
1.CMS (Concurrent Mark Sweep)收集器,常見的B/S架構的應用就適合用這種收集器,因為其高并發(fā),高響應的特點.
CMS收集器是基于“標記-清除”算法實現的,整個收集過程大致分為4個步驟:
初始標記(CMS initial mark)吵护、并發(fā)標記(CMS concurrenr mark)盒音、重新標記(CMS remark)、并發(fā)清除(CMS concurrent sweep)馅而。
初識標記和重新標記會暫停其他線程.但是處理速度很快.整體來說.CMS收集器的內存回收過程是與用戶線程一起并發(fā)執(zhí)行的.
CMS收集器的優(yōu)點:并發(fā)收集祥诽、低停頓,但是CMS還遠遠達不到完美用爪。
CMS收集器主要有三個顯著缺點:
1.對CPU資源非常敏感
2.無法處理浮動垃圾
3.使用 "標記-清除"后,會產生大力量碎片
2.G1收集器原押。相比CMS收集器有不少改進,首先基于標記-整理算法偎血,不會產生內存碎片問題诸衔,其次,可以比較精確的控制停頓
3.Serial Old颇玷。Serial Old是Serial收集器的老年代版本笨农,它同樣使用一個單線程執(zhí)行收集,使用“標記-整理”算法帖渠。主要使用在Client模式下的虛擬機谒亦。
4.Parallel Old。Parallel Old是Parallel Scavenge收集器的老年代版本,使用多線程和“標記-整理”算法份招。
5.RTSJ垃圾收集器切揭,用于Java實時編程
常用垃圾收集算法
① 清除-壓縮
支持壓縮的垃圾回收器(標記-壓縮 = 標記清除+壓縮)會把所有的可達對象搬遷到一端,然后直接清理掉端邊界以外的內存锁摔,減少了內存碎片廓旬。
②標記-清除算法
不壓縮的垃圾回收器(標記-清除)要遍歷兩次,第一次先從跟開始訪問所有可達對象谐腰,并將他們標記為可達狀態(tài)孕豹,第二次便利整個內存區(qū)域,對未標記可達狀態(tài)的對象進行回收處理十气。這種回收方式不壓縮励背,不需要額外內存,但要兩次遍歷砸西,會產生碎片
③復制式的垃圾回收器:將堆內存分成兩個相同空間叶眉,從根(類似于前面的有向圖起始頂點)開始訪問每一個關聯(lián)的可達對象,將空間A的全部可達對象復制到空間B籍胯,然后一次性回收空間A竟闪。對于該算法而言,因為只需訪問所有的可達對象杖狼,將所有的可達對象復制走之后就直接回收整個空間炼蛤,完全不用理會不可達對象,所以遍歷空間的成本較小蝶涩,但需要巨大的復制成本和較多的內存理朋。
4.java分代收集算法
Java主要采用了分代收集算法。分代收集算法主要將對象存活期的長短將內存進行劃分绿聘。
Java主要將內存劃分為兩部分:新生代和老生代
Java的新生代中嗽上,對象的存活率低,存活期期會相對會比較短一些熄攘,所以可以選用復制算法來進行內存回收兽愤。
Java的老生代中,對象的存活率比較高挪圾,并且相對存活期比較長一些浅萧,可以采用標記-清除-壓縮的算法來進行內存回收。
5.內存管理小技巧
1)盡量使用直接量哲思,eg:String javaStr = “小學徒的成長歷程”;
2)使用StringBuilder和StringBuffer進行字符串連接等操作;
3)盡早釋放無用對象;
4)盡量少使用靜態(tài)變量;
5)緩存常用的對象:可以使用開源的開源緩存實現洼畅,eg:OSCache,Ehcache;
6)盡量不使用finalize()方法;
7)在必要的時候可以考慮使用軟引用SoftReference棚赔。