亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > Java > 正文

java 內存分配分析

2019-11-06 06:05:40
字體:
來源:轉載
供稿:網友
本文將由淺入深詳細介紹java內存分配的原理,以幫助新手更輕松的學習Java。這類文章網上有很多,但大多比較零碎。本文從認知過程角度出發,將帶給讀者一個系統的介紹。

進入正題前首先要知道的是Java程序運行在JVM(Java Virtual Machine,Java虛擬機)上,可以把JVM理解成Java程序和操作系統之間的橋梁,JVM實現了Java的平臺無關性,由此可見JVM的重要性。所以在學習Java內存分配原理的時候一定要牢記這一切都是在JVM中進行的,JVM是內存分配原理的基礎與前提。

簡單通俗的講,一個完整的Java程序運行過程會涉及以下內存區域:

寄存器:JVM內部虛擬寄存器,存取速度非???,程序不可控制。

棧:保存局部變量的值,包括:1.用來保存基本數據類型的值;2.保存類的實例,即堆區對象的引用(指針)。也可以用來保存加載方法時的幀。

堆:用來存放動態產生的數據,比如new出來的對象。注意創建出來的對象只包含屬于各自的成員變量,并不包括成員方法。因為同一個類的對象擁有各自的成員變量,存儲在各自的堆中,但是他們共享該類的方法,并不是每創建一個對象就把成員方法復制一次。

常量池:JVM為每個已加載的類型維護一個常量池,常量池就是這個類型用到的常量的一個有序集合。包括直接常量(基本類型,String)和對其他類型、方法、字段的符號引用(1)。池中的數據和數組一樣通過索引訪問。由于常量池包含了一個類型所有的對其他類型、方法、字段的符號引用,所以常量池在Java的動態鏈接中起了核心作用。常量池存在于堆中。

代碼段:用來存放從硬盤上讀取的源程序代碼。

數據段:用來存放static定義的靜態成員。

下面是內存表示圖:

上圖中大致描述了Java內存分配,接下來通過實例詳細講解Java程序是如何在內存中運行的(注:以下圖片引用自尚學堂馬士兵老師的J2SE課件,圖右側是程序代碼,左側是內存分配示意圖,我會一一加上注釋)。

預備知識:

1.一個Java文件,只要有main入口方法,我們就認為這是一個Java程序,可以單獨編譯運行。

2.無論是普通類型的變量還是引用類型的變量(俗稱實例),都可以作為局部變量,他們都可以出現在棧中。只不過普通類型的變量在棧中直接保存它所對應的值,而引用類型的變量保存的是一個指向堆區的指針,通過這個指針,就可以找到這個實例在堆區對應的對象。因此,普通類型變量只在棧區占用一塊內存,而引用類型變量要在棧區和堆區各占一塊內存。

示例:

1.JVM自動尋找main方法,執行第一句代碼,創建一個Test類的實例,在棧中分配一塊內存,存放一個指向堆區對象的指針110925。

2.創建一個int型的變量date,由于是基本類型,直接在棧中存放date對應的值9。

3.創建兩個BirthDate類的實例d1、d2,在棧中分別存放了對應的指針指向各自的對象。他們在實例化時調用了有參數的構造方法,因此對象中有自定義初始值。

調用test對象的change1方法,并且以date為參數。JVM讀到這段代碼時,檢測到i是局部變量,因此會把i放在棧中,并且把date的值賦給i。

把1234賦給i。很簡單的一步。

change1方法執行完畢,立即釋放局部變量i所占用的??臻g。

調用test對象的change2方法,以實例d1為參數。JVM檢測到change2方法中的b參數為局部變量,立即加入到棧中,由于是引用類型的變量,所以b中保存的是d1中的指針,此時b和d1指向同一個堆中的對象。在b和d1之間傳遞是指針。

change2方法中又實例化了一個BirthDate對象,并且賦給b。在內部執行過程是:在堆區new了一個對象,并且把該對象的指針保存在棧中的b對應空間,此時實例b不再指向實例d1所指向的對象,但是實例d1所指向的對象并無變化,這樣無法對d1造成任何影響。

change2方法執行完畢,立即釋放局部引用變量b所占的??臻g,注意只是釋放了??臻g,堆空間要等待自動回收。

調用test實例的change3方法,以實例d2為參數。同理,JVM會在棧中為局部引用變量b分配空間,并且把d2中的指針存放在b中,此時d2和b指向同一個對象。再調用實例b的setDay方法,其實就是調用d2指向的對象的setDay方法。

調用實例b的setDay方法會影響d2,因為二者指向的是同一個對象。

change3方法執行完畢,立即釋放局部引用變量b。

以上就是Java程序運行時內存分配的大致情況。其實也沒什么,掌握了思想就很簡單了。無非就是兩種類型的變量:基本類型和引用類型。二者作為局部變量,都放在棧中,基本類型直接在棧中保存值,引用類型只保存一個指向堆區的指針,真正的對象在堆里。作為參數時基本類型就直接傳值,引用類型傳指針。

小結:

1.分清什么是實例什么是對象。Class a= new Class();此時a叫實例,而不能說a是對象。實例在棧中,對象在堆中,操作實例實際上是通過實例的指針間接操作對象。多個實例可以指向同一個對象。

2.棧中的數據和堆中的數據銷毀并不是同步的。方法一旦結束,棧中的局部變量立即銷毀,但是堆中對象不一定銷毀。因為可能有其他變量也指向了這個對象,直到棧中沒有變量指向堆中的對象時,它才銷毀,而且還不是馬上銷毀,要等垃圾回收掃描時才可以被銷毀。

3.以上的棧、堆、代碼段、數據段等等都是相對于應用程序而言的。每一個應用程序都對應唯一的一個JVM實例,每一個JVM實例都有自己的內存區域,互不影響。并且這些內存區域是所有線程共享的。這里提到的棧和堆都是整體上的概念,這些堆棧還可以細分。

4.類的成員變量在不同對象中各不相同,都有自己的存儲空間(成員變量在堆中的對象中)。而類的方法卻是該類的所有對象共享的,只有一套,對象使用方法的時候方法才被壓入棧,方法不使用則不占用內存。

以上分析只涉及了棧和堆,還有一個非常重要的內存區域:常量池,這個地方往往出現一些莫名其妙的問題。常量池是干嘛的上邊已經說明了,也沒必要理解多么深刻,只要記住它維護了一個已加載類的常量就可以了。接下來結合一些例子說明常量池的特性。

預備知識:

基本類型和基本類型的包裝類?;绢愋陀校篵yte、short、char、int、long、boolean?;绢愋偷陌b類分別是:Byte、Short、Character、Integer、Long、Boolean。注意區分大小寫。二者的區別是:基本類型體現在程序中是普通變量,基本類型的包裝類是類,體現在程序中是引用變量。因此二者在內存中的存儲位置不同:基本類型存儲在棧中,而基本類型包裝類存儲在堆中。上邊提到的這些包裝類都實現了常量池技術,另外兩種浮點數類型的包裝類則沒有實現。另外,String類型也實現了常量池技術。

實例:

[java] view plain copypublic class test {      public static void main(String[] args) {              objPoolTest();      }        public static void objPoolTest() {          int i = 40;          int i0 = 40;          Integer i1 = 40;          Integer i2 = 40;          Integer i3 = 0;          Integer i4 = new Integer(40);          Integer i5 = new Integer(40);          Integer i6 = new Integer(0);          Double d1=1.0;          Double d2=1.0;                    System.out.PRintln("i=i0/t" + (i == i0));          System.out.println("i1=i2/t" + (i1 == i2));          System.out.println("i1=i2+i3/t" + (i1 == i2 + i3));          System.out.println("i4=i5/t" + (i4 == i5));          System.out.println("i4=i5+i6/t" + (i4 == i5 + i6));              System.out.println("d1=d2/t" + (d1==d2));                     System.out.println();              }  }  

 

結果:

[java] view plain copyi=i0    true  i1=i2   true  i1=i2+i3        true  i4=i5   false  i4=i5+i6        true  d1=d2   false  

 

結果分析:

1.i和i0均是普通類型(int)的變量,所以數據直接存儲在棧中,而棧有一個很重要的特性:棧中的數據可以共享。當我們定義了int i = 40;,再定義int i0 = 40;這時候會自動檢查棧中是否有40這個數據,如果有,i0會直接指向i的40,不會再添加一個新的40。

2.i1和i2均是引用類型,在棧中存儲指針,因為Integer是包裝類。由于Integer包裝類實現了常量池技術,因此i1、i2的40均是從常量池中獲取的,均指向同一個地址,因此i1=12。

3.很明顯這是一個加法運算,Java的數學運算都是在棧中進行的,Java會自動對i1、i2進行拆箱操作轉化成整型,因此i1在數值上等于i2+i3。

4.i4和i5均是引用類型,在棧中存儲指針,因為Integer是包裝類。但是由于他們各自都是new出來的,因此不再從常量池尋找數據,而是從堆中各自new一個對象,然后各自保存指向對象的指針,所以i4和i5不相等,因為他們所存指針不同,所指向對象不同。

5.這也是一個加法運算,和3同理。

6.d1和d2均是引用類型,在棧中存儲指針,因為Double是包裝類。但Double包裝類沒有實現常量池技術,因此Doubled1=1.0;相當于Double d1=new Double(1.0);,是從堆new一個對象,d2同理。因此d1和d2存放的指針不同,指向的對象不同,所以不相等。

小結:

1.以上提到的幾種基本類型包裝類均實現了常量池技術,但他們維護的常量僅僅是【-128至127】這個范圍內的常量,如果常量值超過這個范圍,就會從堆中創建對象,不再從常量池中取。比如,把上邊例子改成Integer i1 = 400; Integer i2 = 400;,很明顯超過了127,無法從常量池獲取常量,就要從堆中new新的Integer對象,這時i1和i2就不相等了。

2.String類型也實現了常量池技術,但是稍微有點不同。String型是先檢測常量池中有沒有對應字符串,如果有,則取出來;如果沒有,則把當前的添加進去。

凡是涉及內存原理,一般都是博大精深的領域,切勿聽信一家之言,多讀些文章。我在這只是淺析,里邊還有很多貓膩,就留給讀者探索思考了。希望本文能對大家有所幫助!

腳注:

(1) 符號引用,顧名思義,就是一個符號,符號引用被使用的時候,才會解析這個符號。如果熟悉linux或unix系統的,可以把這個符號引用看作一個文件的軟鏈接,當使用這個軟連接的時候,才會真正解析它,展開它找到實際的文件

對于符號引用,在類加載層面上討論比較多,源碼級別只是一個形式上的討論。

當一個類被加載時,該類所用到的別的類的符號引用都會保存在常量池,實際代碼執行的時候,首次遇到某個別的類時,JVM會對常量池的該類的符號引用展開,轉為直接引用,這樣下次再遇到同樣的類型時,JVM就不再解析,而直接使用這個已經被解析過的直接引用。

除了上述的類加載過程的符號引用說法,對于源碼級別來說,就是依照引用的解析過程來區別代碼中某些數據屬于符號引用還是直接引用,如,System.out.println("test" +"abc");//這里發生的效果相當于直接引用,而假設某個Strings = "abc"; System.out.println("test" + s);//這里的發生的效果相當于符號引用,即把s展開解析,也就相當于s是"abc"的一個符號鏈接,也就是說在編譯的時候,class文件并沒有直接展看s,而把這個s看作一個符號,在實際的代碼執行時,才會展開這個。

參考文章:

java內存分配研究:http://www.blogjava.net/Jack2007/archive/2008/05/21/202018.html

Java常量池詳解之一道比較蛋疼的面試題:http://www.cnblogs.com/DreamSea/archive/2011/11/20/2256396.html

jvm常量池:http://www.cnblogs.com/wenfeng762/archive/2011/08/14/2137820.html

深入Java核心 Java內存分配原理精講:http://developer.51cto.com/art/201009/225071.htm

 


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
98视频在线噜噜噜国产| 久久综合88中文色鬼| 懂色av中文一区二区三区天美| 国产日韩中文字幕在线| 日韩在线视频观看| 亚洲人成电影网站色…| 成人444kkkk在线观看| 91av视频在线播放| 欧美极品少妇xxxxx| 91精品国产91久久久久久不卡| 欧美成人免费va影院高清| 欧美午夜精品久久久久久久| 欧美日韩国产色| 精品国产电影一区| 91久久精品国产| 中文字幕国产精品久久| 国产成人精品久久二区二区| 懂色aⅴ精品一区二区三区蜜月| 91精品久久久久久久久青青| 欧美成人一区在线| 奇米成人av国产一区二区三区| 欧美wwwwww| 黄色成人av网| 欧美成人免费一级人片100| 欧美性猛交xxxxx水多| 亚洲欧美在线看| 精品中文字幕在线2019| 91黑丝在线观看| 日韩中文字幕亚洲| 久久久天堂国产精品女人| 中文字幕亚洲情99在线| 亚洲成人激情图| 久久久久成人精品| 亚洲精品v天堂中文字幕| 欧美午夜精品久久久久久久| 91久久嫩草影院一区二区| 国产成人精彩在线视频九色| 欧美精品第一页在线播放| 久久这里有精品| 疯狂蹂躏欧美一区二区精品| 久久亚洲精品国产亚洲老地址| 亚洲aa中文字幕| 亚洲人成电影网站色| 欧美电影免费观看网站| 欧美在线观看一区二区三区| 欧美日韩第一页| 91爱爱小视频k| 日韩成人高清在线| 欧美视频在线看| 中文字幕在线日韩| 国产黑人绿帽在线第一区| 日韩www在线| 欧美性猛交xxxx富婆| 中文字幕日本精品| 国产成人短视频| 美女黄色丝袜一区| 亚洲深夜福利在线| 亚洲精品国产精品国自产在线| 色婷婷亚洲mv天堂mv在影片| 成人欧美一区二区三区黑人孕妇| 国产精品99免视看9| 这里只有精品丝袜| 成人h片在线播放免费网站| 亚洲一区二区在线| 欧美激情图片区| 久久精品国产综合| 日韩在线视频播放| 欧美激情奇米色| 国产欧美一区二区白浆黑人| 成人h猎奇视频网站| 波霸ol色综合久久| 成人免费福利视频| 性欧美办公室18xxxxhd| 欧美激情喷水视频| 精品久久中文字幕| 精品久久久久久电影| 国产一区二区黄| 在线精品国产欧美| 日韩欧美a级成人黄色| 国产一区二区三区免费视频| 91久久国产精品| 亚洲精选在线观看| 中文字幕欧美日韩va免费视频| 亚洲精品国偷自产在线99热| 国产精品美女www| 日韩欧美中文字幕在线播放| 亚洲精品91美女久久久久久久| 国模极品一区二区三区| 高清在线视频日韩欧美| 久久99视频精品| www.日韩视频| 92版电视剧仙鹤神针在线观看| 亚洲国产精品电影在线观看| 日韩成人性视频| 懂色aⅴ精品一区二区三区蜜月| 亚洲va电影大全| 欧美日韩国产丝袜另类| 国产精品久久久久久久美男| 久久偷看各类女兵18女厕嘘嘘| 欧美视频在线观看免费网址| 日韩欧美中文字幕在线观看| 超碰日本道色综合久久综合| 夜色77av精品影院| 91精品在线观| 色妞在线综合亚洲欧美| 亚洲mm色国产网站| 国产精品中文在线| 国产69精品久久久久9999| 亚洲欧美精品在线| 美女久久久久久久久久久| 日韩免费在线免费观看| 不卡av电影在线观看| 国产精品视频免费观看www| 成人亲热视频网站| 欧美极品欧美精品欧美视频| 国产欧美va欧美va香蕉在| 国产精品欧美激情在线播放| 91精品国产综合久久香蕉的用户体验| 成人激情视频在线观看| 最近2019年手机中文字幕| 日韩亚洲精品电影| 欧美电影免费看| 成人免费网站在线| 91在线视频导航| 亚洲最大的av网站| 欧美国产日韩一区| 亚洲国产私拍精品国模在线观看| 2021久久精品国产99国产精品| 最近2019中文字幕大全第二页| 日韩av不卡在线| 亚洲成**性毛茸茸| **欧美日韩vr在线| 欧美野外wwwxxx| 欧美日韩一区二区免费视频| 国产精品扒开腿爽爽爽视频| 国产精品国产三级国产aⅴ浪潮| 国产精品jvid在线观看蜜臀| 91九色视频导航| 国产精品久久久亚洲| 亚洲a区在线视频| 日韩福利视频在线观看| 国产精品h在线观看| 亚洲三级av在线| 国产成人短视频| 欧美老女人性生活| 成人免费看黄网站| 色999日韩欧美国产| 97在线视频一区| 日韩精品福利网站| www.欧美精品一二三区| 精品人伦一区二区三区蜜桃免费| 国产精品国产亚洲伊人久久| 国产成人精品免费久久久久| 69av在线播放| 亚洲精品在线不卡| 国产97人人超碰caoprom| 亚洲国产精品久久久久久| 欧美在线视频a| 亚洲第一黄色网| 成人欧美在线观看| 亚洲美女av网站| 久久国产精品久久精品| 国产欧美精品日韩精品| 97av在线视频免费播放|