溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Java虛擬機(jī)內(nèi)存優(yōu)化的方法

發(fā)布時(shí)間:2021-07-21 22:16:44 來源:億速云 閱讀:236 作者:chen 欄目:編程語言

這篇文章主要介紹“Java虛擬機(jī)內(nèi)存優(yōu)化的方法”,在日常操作中,相信很多人在Java虛擬機(jī)內(nèi)存優(yōu)化的方法問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”Java虛擬機(jī)內(nèi)存優(yōu)化的方法”的疑惑有所幫助!接下來,請跟著小編一起來學(xué)習(xí)吧!

前面一篇文章介紹了Java虛擬機(jī)的體系結(jié)構(gòu)和內(nèi)存模型,既然提到內(nèi)存,就不得不說到內(nèi)存泄露。眾所周知,Java是從C++的基礎(chǔ)上發(fā)展而來的,而C++程序的很大的一個問題就是內(nèi)存泄露難以解決,盡管Java的JVM有一套自己的垃圾回收機(jī)制來回收內(nèi)存,在許多情況下并不需要java程序開發(fā)人員操太多的心,但也是存在泄露問題的,只是比C++小一點(diǎn)。比如說,程序中存在被引用但無用的對象:程序引用了該對象,但后續(xù)不會或者不能再使用它,那么它占用的內(nèi)存空間就浪費(fèi)了。

我們先來看看GC是如何工作的:監(jiān)控每一個對象的運(yùn)行狀態(tài),包括對象的申請、引用、被引用、賦值等,當(dāng)該對象不再被引用時(shí),釋放對象(GC本文的重點(diǎn),不做過多闡述)。很多Java程序員過分依賴GC,但問題的關(guān)鍵是無論JVM的垃圾回收機(jī)制做得多好,內(nèi)存總歸是有限的資源,因此就算GC會為我們完成了大部分的垃圾回收,但適當(dāng)?shù)刈⒁饩幋a過程中的內(nèi)存優(yōu)化還是很必要的。這樣可以有效的減少GC次數(shù),同時(shí)提升內(nèi)存利用率,***限度地提高程序的效率。

總體而言,Java虛擬機(jī)的內(nèi)存優(yōu)化應(yīng)從兩方面著手:Java虛擬機(jī)和Java應(yīng)用程序。前者指根據(jù)應(yīng)用程序的設(shè)計(jì)通過虛擬機(jī)參數(shù)控制虛擬機(jī)邏輯內(nèi)存分區(qū)的大小以使虛擬機(jī)的內(nèi)存與程序?qū)?nèi)存的需求相得益彰;后者指優(yōu)化程序算法,降低GC負(fù)擔(dān),提高GC回收成功率。

通過參數(shù)優(yōu)化虛擬機(jī)內(nèi)存的參數(shù)如下所示:

◆  Xms

初始Heap大小

◆ Xmx

java heap***值

◆ Xmn

young generation的heap大小

◆ Xss

每個線程的Stack大小

上面是三個比較常用的參數(shù),還有一些:

◆ XX:MinHeapFreeRatio=40

Minimum percentage of heap free after GC to avoid expansion.

◆ XX:MaxHeapFreeRatio=70

Maximum percentage of heap free after GC to avoid shrinking.

◆ XX:NewRatio=2

Ratio of new/old generation sizes. [Sparc -client:8; x86 -server:8; x86 -client:12.]-client:8 (1.3.1+), x86:12]

◆ XX:NewSize=2.125m

Default size of new generation (in bytes) [5.0 and newer: 64 bit VMs are scaled 30% larger; x86:1m; x86, 5.0 and older: 640k]

◆ XX:MaxNewSize=

Maximum size of new generation (in bytes). Since 1.4, MaxNewSize is computed as a function of NewRatio.

◆ XX:SurvivorRatio=25

Ratio of eden/survivor space size [Solaris amd64: 6; Sparc in 1.3.1: 25; other Solaris platforms in 5.0 and earlier: 32]

◆ XX:PermSize=

Initial size of permanent generation

◆ XX:MaxPermSize=64m

Size of the Permanent Generation. [5.0 and newer: 64 bit VMs are scaled 30% larger; 1.4 amd64: 96m; 1.3.1 -client: 32m.]

下面所說通過優(yōu)化程序算法來提高內(nèi)存利用率,并降低內(nèi)存風(fēng)險(xiǎn),完全是經(jīng)驗(yàn)之談,僅供參考,如有不妥,請指正,謝謝!

1.盡早釋放無用對象的引用(XX = null;)

看一段代碼:

public List<PageData> parse(HtmlPage page) {          List<PageData> list = null;                  try {              List valueList = page.getByXPath(config.getContentXpath());              if (valueList == null || valueList.isEmpty()) {                  return list;              }              //需要時(shí)才創(chuàng)建對象,節(jié)省內(nèi)存,提高效率              list = new ArrayList<PageData>();              PageData pageData = new PageData();              StringBuilder value = new StringBuilder();              for (int i = 0; i < valueList.size(); i++) {                  HtmlElement content = (HtmlElement) valueList.get(i);                  DomNodeList<HtmlElement> imgs = content.getElementsByTagName("img");                  if (imgs != null && !imgs.isEmpty()) {                      for (HtmlElement img : imgs) {                          try {                              HtmlImage image = (HtmlImage) img;                              String path = image.getSrcAttribute();                              String format = path.substring(path.lastIndexOf("."), path.length());                              String localPath = "D:/images/" + MD5Helper.md5(path).replace("\\", ",").replace("/", ",") + format;                              File localFile = new File(localPath);                              if (!localFile.exists()) {                                  localFile.createNewFile();                                  image.saveAs(localFile);                              }                              image.setAttribute("src", "file:///" + localPath);                              localFile = null;                              image = null;                              img = null;                          } catch (Exception e) {                          }                      }                      //這個對象以后不會在使用了,清除對其的引用,等同于提前告知GC,該對象可以回收了                      imgs = null;                  }                  String text = content.asXml();                  value.append(text).append("<br/>");                  valueList=null;                  content = null;                  text = null;              }              pageData.setContent(value.toString());              pageData.setCharset(page.getPageEncoding());                         list.add(pageData);              //這里 pageData=null; 是沒用的,因?yàn)閘ist仍然持有該對象的引用,GC不會回收它              value=null;              //這里可不能 list=null; 因?yàn)閘ist是方法的返回值,否則你從該方法中得到的返回值永遠(yuǎn)為空,而且這種錯誤不易被發(fā)現(xiàn)、排除          } catch (Exception e) {                      }                  return list;  }

2.謹(jǐn)慎使用集合數(shù)據(jù)類型,如數(shù)組,樹,圖,鏈表等數(shù)據(jù)結(jié)構(gòu),這些數(shù)據(jù)結(jié)構(gòu)對GC來說回收更復(fù)雜。

3.避免顯式申請數(shù)組空間,不得不顯式申請時(shí),盡量準(zhǔn)確估計(jì)其合理值。

4.盡量避免在類的默認(rèn)構(gòu)造器中創(chuàng)建、初始化大量的對象,防止在調(diào)用其自類的構(gòu)造器時(shí)造成不必要的內(nèi)存資源浪費(fèi)

5.盡量避免強(qiáng)制系統(tǒng)做垃圾內(nèi)存的回收,增長系統(tǒng)做垃圾回收的最終時(shí)間

6.盡量做遠(yuǎn)程方法調(diào)用類應(yīng)用開發(fā)時(shí)使用瞬間值變量,除非遠(yuǎn)程調(diào)用端需要獲取該瞬間值變量的值。

7.盡量在合適的場景下使用對象池技術(shù)以提高系統(tǒng)性能

到此,關(guān)于“Java虛擬機(jī)內(nèi)存優(yōu)化的方法”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實(shí)踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識,請繼續(xù)關(guān)注億速云網(wǎng)站,小編會繼續(xù)努力為大家?guī)砀鄬?shí)用的文章!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI