本文目錄一覽:
Java的內存分配是什麼樣的
關於java內存分配是這樣的:
程序代碼存儲在”code segment”中,靜態變量和字符串常量存儲在「data segment”區域中,局部變量存儲在”stack”(棧內存)中,nwe出來的東西(即對象)存儲在”heap”(堆內存)中
Java內存劃分到底是4個部分還是5個部分?
Java把內存劃分成兩種:一種是棧內存,一種是堆內存。在函數中定義的一些基本類型的變量和對象的引用變量都在函數的棧內存中分配。當在一段代碼塊定義一個變量時,Java就在棧中為這個變量分配內存空間,當超過變量的作用域後,Java會自動釋放掉為該變量所分配的內存空間,該內存空間可以立即被另作他用。堆內存用來存放由new創建的對象和數組。在堆中分配的內存,由Java虛擬機的自動垃圾回收器來管理。在堆中產生了一個數組或對象後,還可以在棧中定義一個特殊的變量,讓棧中這個變量的取值等於數組或對象在堆內存中的首地址,棧中的這個變量就成了數組或對象的引用變量。引用變量就相當於是為數組或對象起的一個名稱,以後就可以在程序中使用棧中的引用變量來訪問堆中的數組或對象。具體的說:棧與堆都是Java用來在Ram中存放數據的地方。與C++不同,Java自動管理棧和堆,程序員不能直接地設置棧或堆。Java的堆是一個運行時數據區,類的(對象從中分配空間。這些對象通過new、newarray、anewarray和multianewarray等指令建立,它們不需要程序代碼來顯式的釋放。堆是由垃圾回收來負責的,堆的優勢是可以動態地分配內存大小,生存期也不必事先告訴編譯器,因為它是在運行時動態分配內存的,Java的垃圾收集器會自動收走這些不再使用的數據。但缺點是,由於要在運行時動態分配內存,存取速度較慢。棧的優勢是,存取速度比堆要快,僅次於寄存器,棧數據可以共享。但缺點是,存在棧中的數據大小與生存期必須是確定的,缺乏靈活性。棧中主要存放一些基本類型的變量(,int,short,long,byte,float,double,boolean,char)和對象句柄。棧有一個很重要的特殊性,就是存在棧中的數據可以共享。假設我們同時定義:inta=3;intb=3;編譯器先處理inta=3;首先它會在棧中創建一個變量為a的引用,然後查找棧中是否有3這個值,如果沒找到,就將3存放進來,然後將a指向3。接着處理intb=3;在創建完b的引用變量後,因為在棧中已經有3這個值,便將b直接指向3。這樣,就出現了a與b同時均指向3的情況。這時,如果再令a=4;那麼編譯器會重新搜索棧中是否有4值,如果沒有,則將4存放進來,並令a指向4;如果已經有了,則直接將a指向這個地址。因此a值的改變不會影響到b的值。要注意這種數據的共享與兩個對象的引用同時指向一個對象的這種共享是不同的,因為這種情況a的修改並不會影響到b,它是由編譯器完成的,它有利於節省空間。而一個對象引用變量修改了這個對象的內部狀態,會影響到另一個對象引用變量。
JAVA里,ArrayList在內存分配上究竟是怎樣的?
ArrayList就是動態數組,也是一個對象。
創建一個ArrayList對象,該對象存放在堆內存中,且是一個內存連續的內存區域。
1、ArrayList是用數組實現的,這個數組的內存是連續的,不存在相鄰元素之間還隔着其他內存。
2、索引ArrayList時,速度比原生數組慢是因為你要用get方法,這是一個函數調用,而數組直接用[ ]訪問,相當於直接操作內存地址,速度當然比函數調用快。
3、新建ArrayList的時候,JVM為其分配一個默認或指定大小的連續內存區域(封裝為數組)。
4、每次增加元素會檢查容量,不足則創建新的連續內存區域(大小等於初始大小+步長),也用數組形式封裝,並將原來的內存區域數據複製到新的內存區域,然後再用ArrayList中引用原來封裝的數組對象的引用變量引用到新的數組對象:
elementData = Arrays.copyOf(elementData, newCapacity);
5、集合內存分配以及初始化過程圖解
原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-hk/n/257558.html