<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    jialisoftw

    JAVA中的內存溢出

    一、概述
    內存溢出與數據庫鎖表的問題,可以說是開發人員的噩夢,一般的程序異常,總是可以知道在什么時候或是在什么操作步驟上出現了異常,而且根據堆棧信息也很容易定位到程序中是某處出現了問題。內存溢出與鎖表則不然,一般現象是操作一般時間后系統越來越慢,直到死機,但并不能明確是在什么操作上出現的,發生的時間點也沒有規律,查看日志或查看數據庫也不能定位出問題的代碼。
    更嚴重的是內存溢出與數據庫鎖表在系統開發和單元測試階段并不容易被發現,當系統正式上線一般時間后,操作的并發量上來了,數據也積累了一些,系統就容易出現內存溢出或是鎖表的現象,而此時系統又不能隨意停機或重啟,為修正BUG帶來很大的困難。本文以筆者開發和支持的多個項目為例,與大家分享在開發過程中遇到的Java內存溢出和數據庫鎖表的檢測和處理解決過程。
    二、內存溢出的分析
    內存溢出是指應用系統中存在無法回收的內存或使用的內存過多,最終使得程序運行要用到的內存大于虛擬機能提供的最大內存。為了解決Java中內存溢出問題,首先必須了解Java是如何管理內存的。Java的內存管理就是對象的分配和釋放問題。
    在Java中,內存的分配是由程序完成的,而內存的釋放是由垃圾收集器(Garbage Collection,GC)完成的,程序員不需要通過調用GC函數來釋放內存,因為不同的JVM實現者可能使用不同的算法管理GC,有的是內存使用到達一定程度時,GC才開始工作,也有定時執行的,有的是中斷式執行GC。但GC只能回收無用并且不再被其它對象引用的那些對象所占用的空間。Java的內存垃圾回收機制是從程序的主要運行對象開始檢查引用鏈,當遍歷一遍后發現沒有被引用的孤立對象就作為垃圾回收。引起內存溢出的原因有很多種,常見的有以下幾種:
    1 內存中加載的數據量過于龐大,如一次從數據庫取出過多數據;
    2 集合類中有對對象的引用,使用完后未清空,使得JVM不能回收;
    3 代碼中存在死循環或循環產生過多重復的對象實體;
    4 使用的第三方軟件中的BUG;
    5 啟動參數內存值設定的過小;
    三、內存溢出的解決
    內存溢出雖然很棘手,但也有相應的解決辦法,可以按照從易到難,一步步的解決。
    第一步,就是修改JVM啟動參數,直接增加內存。這一點看上去似乎很簡單,但很容易被忽略。JVM默認可以使用的內存為64M,Tomcat默認可以使用的內存為128MB,對于稍復雜一點的系統就會不夠用。在某項目中,就因為啟動參數使用的默認值,經常報“OutOfMemory”錯誤。因此,-Xms,-Xmx參數一定不要忘記加。
    第二步,檢查錯誤日志,查看“OutOfMemory”錯誤前是否有其它異常或錯誤。在一個項目中,使用兩個數據庫連接,其中專用于發送短信的數據庫連接使用DBCP連接池管理,用戶為不將短信發出,有意將數據庫連接用戶名改錯,使得日志中有許多數據庫連接異常的日志,一段時間后,就出現 “OutOfMemory”錯誤。經分析,這是由于DBCP連接池BUG引起的,數據庫連接不上后,沒有將連接釋放,最終使得DBCP報 “OutOfMemory”錯誤。經過修改正確數據庫連接參數后,就沒有再出現內存溢出的錯誤。查看日志對于分析內存溢出是非常重要的,通過仔細查看日志,分析內存溢出前做過哪些操作,可以大致定位有問題的模塊。
    第三步,安排有經驗的編程人員對代碼進行走查和分析,找出可能發生內存溢出的位置。重點排查以下幾點:
    (1) 檢查代碼中是否有死循環或遞歸調用。
    (2) 檢查是否有大循環重復產生新對象實體。
    (3) 檢查對數據庫查詢中,是否有一次獲得全部數據的查詢。一般來說,如果一次取十萬條記錄到內存,就可能引起內存溢出。這個問題比較隱蔽,在上線前,數據庫中數據較少,不容易出問題,上線后,數據庫中數據多了,一次查詢就有可能引起內存溢出。因此對于數據庫查詢盡量采用分頁的方式查詢。
    (4) 檢查List、MAP等集合對象是否有使用完后,未清除的問題。List、MAP等集合對象會始終存有對對象的引用,使得這些對象不能被GC回收。
    這里引用一個常看到的例子,在下面的代碼中,循環申請Object對象,并將所申請的對象放入一個Vector中,如果僅僅釋放對象本身,但因為Vector仍然引用該對象,所以這個對象對GC來說是不可回收的。因此如果對象加入到Vector后,還必須從Vector中刪除,最簡單的方法就是將Vector對象設置為null。實際上這些對象已經是無用的,但還被引用,GC就無能為力了(事實上GC認為它還有用),這一點是導致內存泄漏最重要的原因。
    Vector v = new Vector(10);     
    for (int i = 1; i < 100; i++)     
    {     
    Object o = new Object();     
    v.add(o);     
    o = null;     
    } // 此時所有的Object對象都沒有被釋放,因為變量v引用這些對象
    再引用另一個例子來說明Java的內存泄漏。假設有一個日志類Logger,其提供一個靜態的log(String msg),任何其它類都可以調用Logger.Log(message)來將message的內容記錄到系統的日志文件中。Logger類有一個類型為HashMap的靜態變量temp,每次在執行log(message)的時候,都首先將message的值寫入temp中(以當前線程+當前時間為鍵),在退出之前再從temp中將以當前線程和當前時間為鍵的條目刪除。注意,這里當前時間是不斷變化的,所以log在退出之前執行刪除條目的操作并不能刪除執行之初寫入的條目。這樣,任何一個作為參數傳給log的字符串最終由于被Logger的靜態變量temp引用,而無法得到回收,這種對象保持就是我們所說的Java內存泄漏。總的來說,內存管理中的內存泄漏產生的主要原因:保留下來卻永遠不再使用的對象引用。
    第四步,使用內存查看工具動態查看內存使用情況。某個項目上線后,每次系統啟動兩天后,就會出現內存溢出的錯誤。這種情況一般是代碼中出現了緩慢的內存泄漏,用上面三個步驟解決不了,這就需要使用內存查看工具了。
    內存查看工具有許多,比較有名的有:Optimizeit Profiler、JProbe Profiler、JinSight和Java1.5的Jconsole等。它們的基本工作原理大同小異,都是監測Java程序運行時所有對象的申請、釋放等動作,將內存管理的所有信息進行統計、分析、可視化。開發人員可以根據這些信息判斷程序是否有內存泄漏問題。一般來說,一個正常的系統在其啟動完成后其內存的占用量是基本穩定的,而不應該是無限制的增長的。持續地觀察系統運行時使用的內存的大小,可以看到在內存使用監控窗口中是基本規則的鋸齒形的圖線,如果內存的大小持續地增長,則說明系統存在內存泄漏問題。通過間隔一段時間取一次內存快照,然后對內存快照中對象的使用與引用等信息進行比對與分析,可以找出是哪個類的對象在泄漏。
    四、總結
    通過以上四個步驟的分析與處理,基本能處理內存溢出的問題。當然,在這些過程中也需要相當的經驗與敏感度,需要在實際的開發與調試過程中不斷積累。總體上來說,產生內存溢出是由于代碼寫的不好造成的,因此提高代碼的質量是最根本的解決辦法。有的人認為先把功能實現,有BUG時再在測試階段進行修正,這種想法是錯誤的。正如一件產品的質量是在生產制造的過程中決定的,而不是質量檢測時決定的,軟件的質量在設計與編碼階段就已經決定了,測試只是對軟件質量的一個驗證,因為測試不可能找出軟件中所有的BUG。

    posted on 2013-05-03 10:26 飛豬一號 閱讀(293) 評論(0)  編輯  收藏


    只有注冊用戶登錄后才能發表評論。


    網站導航:
     

    導航

    <2013年5月>
    2829301234
    567891011
    12131415161718
    19202122232425
    2627282930311
    2345678

    統計

    常用鏈接

    留言簿

    隨筆檔案

    友情鏈接

    搜索

    最新評論

    閱讀排行榜

    評論排行榜

    主站蜘蛛池模板: 成人免费av一区二区三区| 亚洲s码欧洲m码吹潮| 韩国免费A级毛片久久| 红杏亚洲影院一区二区三区| 国产精品视频全国免费观看| 亚洲一区二区视频在线观看| 人成电影网在线观看免费| 久99精品视频在线观看婷亚洲片国产一区一级在线 | 日韩精品无码免费一区二区三区 | 深夜久久AAAAA级毛片免费看| 免费在线精品视频| 一区二区三区免费视频网站| 亚洲综合无码精品一区二区三区 | 亚洲色图国产精品| 日本高清在线免费| 亚洲久热无码av中文字幕| 免费国产在线观看不卡| 成全视成人免费观看在线看| 无码久久精品国产亚洲Av影片| 1000部羞羞禁止免费观看视频| 亚洲综合一区二区三区四区五区| 日韩免费视频播播| 99视频在线观看免费| 亚洲福利视频网址| 免费大学生国产在线观看p| aaa毛片免费观看| 亚洲免费电影网站| 免费成人av电影| 日本免费高清视频| 亚洲欧洲专线一区| 亚洲中文字幕无码一区二区三区| 69av免费观看| 特级一级毛片免费看| 久久亚洲AV无码精品色午夜麻豆| 国产福利免费在线观看| 久久久精品午夜免费不卡| 亚洲熟女www一区二区三区| 亚洲一区二区三区自拍公司| 毛片免费在线观看网站| 国产一二三四区乱码免费| 亚洲综合色区中文字幕|