當前位置:首頁 » 服務存儲 » 堆存儲的內容
擴展閱讀
webinf下怎麼引入js 2023-08-31 21:54:13
堡壘機怎麼打開web 2023-08-31 21:54:11

堆存儲的內容

發布時間: 2022-09-01 10:22:20

『壹』 Java堆中到底存放些什麼

當Java程序創建一個類的實例或者數組時,都在堆中為新的對象分配內存。虛擬機中只有一個堆,所有的線程都共享他。Java中所有的對象都存放在堆中,包括class對象和異常對象。 那麼這些對象中有存放些什麼呢?實例數據是肯定的,還有就是當通過對象訪問類信息時就必須有一個指針將對象和方法區中的類信息關聯起來,關聯的方法有多種。一個可能的堆的設計是將堆分為兩個部分:引用池和對象池。一個對象的引用就是指向引用池的本地指針。每一個引用池中的條目都包含兩個部分:指向對象池中對 象數據的指針和方法區中對象類數據的指針。這種設計能夠方便Java虛擬機堆碎片的整理。當虛擬機在對象池中移動一個對象的時候,只需要修改對應引用池中 的指針地址。但是每次訪問對象的數據都需要處理兩次指針。下圖演示了這種堆的設計。 另一種堆的設計是:一個對象的引用就是一個指向一堆數據和指向相應對象的偏移指針。這種設計方便了對象的訪問,可是對象的移動要變的異常復雜。下圖演示了這種設計 無論虛擬機實現者使用哪一種設計,他都可能為每一個對象保存一個類似方法列表的信息。因為他可以提升對象方法調用的速度,對提升虛擬機的性能非常重要,但 是虛擬機的規范中比沒有要求必須實現類似的數據結構。下圖描述了這種結構。圖中顯示了一個對象引用相關聯的所有的數據結構,包括: 1)、一個指向類型數據的指針 2)、一個對象的方法列表。方法列表是一個指向所有可能被調用對象方法的指針數組。方法數據包括三個部分:操作碼堆棧的大小和方法堆棧的本地變數區;方法的位元組碼;異常列表。 除此之外,堆上的對象數據還有一種邏輯部分,那就是對象鎖,這是一個互斥對象。虛擬機中的每個對象都有一個對象鎖,它被用於協調多個線程訪問同一個對象時的同步。只有當第一次需要加鎖的時候才分配對應的鎖數據,但這時虛擬機需要用某種間接方法來聯系對象數據和對應的鎖數據。這也是為什麼很多對象在其整個生命周期內都沒有被任何線程加鎖。除了實現鎖所需要的數據外,每個Java對象邏輯上還與實現等待集合(wait set)相關聯。 最後一種數據類型-是與垃圾收集器有關的數據。垃圾收集器必須以某種方式跟蹤程序引用的每個對象,這個任務不可避免的要附加一些數據給這些對象。

『貳』 java 堆和棧 分別存什麼東西

棧上不會存儲方法,方法是隨對象變化的,內存給棧分配的內存很小,
java中在內存中可以分為3大部分棧,堆,靜態區,方法作為對象的一部分存儲在堆中,堆中還包含池(用來存儲String類型)。

『叄』 JVM的堆中到底存放的是什麼 是一個個具體的對象信息,還是真實表示對象的Class對象。

成員屬性是用來組成的對象的,既然對象在堆空間中存在,那麼他的成員屬性當然也是在堆中存放著的,又因為age是int這種primate
type的所以堆中直接存放著age的值。

『肆』 堆和棧分別用於保存什麼

棧區(stack)― 由編譯器自動分配釋放 ,存放函數的參數值,局部變數的值等。其操作方式類似於數據結構中的棧。

堆區(heap) ― 一般由程序員分配釋放, 若程序員不釋放,程序結束時可能由OS回收 。注意它與數據結構中的堆是兩回事,分配方式倒是類似於鏈表,呵呵。

『伍』 堆和棧的基本理論知識

堆棧 網路名片
在計算機領域,堆棧是一個不容忽視的概念,但是很多人甚至是計算機專業的人也沒有明確堆棧其實是兩種數據結構。堆棧都是一種數據項按序排列的數據結構,只能在一端(稱為棧頂(top))對數據項進行插入和刪除。要點:堆:順序隨意棧:後進先出(Last-In/First-Out)
目錄[隱藏]

堆和棧的區別
堆和棧的理論知識1.申請方式
2.申請後系統的響應
3.申請大小的限制
4.申請效率的比較
5.堆和棧中的存儲內容
6.存取效率的比較
7.小結:
堆和棧的主要分別:
補充堆和棧的區別
堆和棧的理論知識 1.申請方式
2.申請後系統的響應
3.申請大小的限制
4.申請效率的比較
5.堆和棧中的存儲內容
6.存取效率的比較
7.小結:
堆和棧的主要分別:
補充

[編輯本段]堆和棧的區別
一、預備知識—程序的內存分配 一個由C/C++編譯的程序佔用的內存分為以下幾個部分 1、棧區(stack)— 由編譯器自動分配釋放 ,存放函數的參數值,局部變數的值等。其操作方式類似於數據結構中的棧。 2、堆區(heap) — 一般由程序員分配釋放, 若程序員不釋放,程序結束時可能由OS回收 。注意它與數據結構中的堆是兩回事,分配方式倒是類似於鏈表。 3、全局區(靜態區)(static)— 全局變數和靜態變數的存儲是放在一塊的,初始化的全局變數和靜態變數在一塊區域, 未初始化的全局變數和未初始化的靜態變數在相鄰的另一塊區域。程序結束後由系統釋放。 4、文字常量區 — 常量字元串就是放在這里的,程序結束後由系統釋放 。 5、程序代碼區 — 存放函數體的二進制代碼。 二、例子程序 這是一個前輩寫的,非常詳細 //main.cpp int a = 0; 全局初始化區 char *p1; 全局未初始化區 main() { int b; 棧 char s[] = "abc"; 棧 char *p2; 棧 char *p3 = "123456"; 123456\0在常量區,p3在棧上。 static int c =0; 全局(靜態)初始化區 p1 = (char *)malloc(10); p2 = (char *)malloc(20); } 分配得來得10和20位元組的區域就在堆區。 strcpy(p1, "123456"); 123456\0放在常量區,編譯器可能會將它與p3所指向的"123456"優化成一個地方。
[編輯本段]堆和棧的理論知識

1.申請方式
stack: 由系統自動分配。 例如,聲明在函數中一個局部變數 int b; 系統自動在棧中為b開辟空間 heap: 需要程序員自己申請,並指明大小,在c中malloc函數 如p1 = (char *)malloc(10); 在C++中用new運算符 如p2 = new char[20];//(char *)malloc(10); 但是注意p1、p2本身是在棧中的。
2.申請後系統的響應
棧:只要棧的剩餘空間大於所申請空間,系統將為程序提供內存,否則將報異常提示棧溢出。 堆:首先應該知道操作系統有一個記錄空閑內存地址的鏈表,當系統收到程序的申請時,會遍歷該鏈表,尋找第一個空間大於所申請空間的堆結點,然後將該結點從空閑結點鏈表中刪除,並將該結點的空間分配給程序,另外,對於大多數系統,會在這塊內存空間中的首地址處記錄本次分配的大小,這樣,代碼中的delete語句才能正確的釋放本內存空間。另外,由於找到的堆結點的大小不一定正好等於申請的大小,系統會自動的將多餘的那部分重新放入空閑鏈表中。
3.申請大小的限制
棧:在Windows下,棧是向低地址擴展的數據結構,是一塊連續的內存的區域。這句話的意思是棧頂的地址和棧的最大容量是系統預先規定好的,在 WINDOWS下,棧的大小是2M(也有的說是1M,總之是一個編譯時就確定的常數),如果申請的空間超過棧的剩餘空間時,將提示overflow。因此,能從棧獲得的空間較小。 堆:堆是向高地址擴展的數據結構,是不連續的內存區域。這是由於系統是用鏈表來存儲的空閑內存地址的,自然是不連續的,而鏈表的遍歷方向是由低地址向高地址。堆的大小受限於計算機系統中有效的虛擬內存。由此可見,堆獲得的空間比較靈活,也比較大。
4.申請效率的比較
棧由系統自動分配,速度較快。但程序員是無法控制的。 堆是由new分配的內存,一般速度比較慢,而且容易產生內存碎片,不過用起來最方便. 另外,在WINDOWS下,最好的方式是用VirtualAlloc分配內存,他不是在堆,也不是在棧,而是直接在進程的地址空間中保留一塊內存,雖然用起來最不方便。但是速度快,也最靈活
5.堆和棧中的存儲內容
棧: 在函數調用時,第一個進棧的是主函數中函數調用後的下一條指令(函數調用語句的下一條可執行語句)的地址,然後是函數的各個參數,在大多數的C編譯器中,參數是由右往左入棧的,然後是函數中的局部變數。注意靜態變數是不入棧的。 當本次函數調用結束後,局部變數先出棧,然後是參數,最後棧頂指針指向最開始存的地址,也就是主函數中的下一條指令,程序由該點繼續運行。 堆:一般是在堆的頭部用一個位元組存放堆的大小。堆中的具體內容有程序員安排。
6.存取效率的比較
char s1[] = "aaaaaaaaaaaaaaa"; char *s2 = "bbbbbbbbbbbbbbbbb"; aaaaaaaaaaa是在運行時刻賦值的; 而bbbbbbbbbbb是在編譯時就確定的; 但是,在以後的存取中,在棧上的數組比指針所指向的字元串(例如堆)快。 比如: #include void main() { char a = 1; char c[] = "1234567890"; char *p ="1234567890"; a = c[1]; a = p[1]; return; } 對應的匯編代碼 10: a = c[1]; 00401067 8A 4D F1 mov cl,byte ptr [ebp-0Fh] 0040106A 88 4D FC mov byte ptr [ebp-4],cl 11: a = p[1]; 0040106D 8B 55 EC mov edx,dword ptr [ebp-14h] 00401070 8A 42 01 mov al,byte ptr [edx+1] 00401073 88 45 FC mov byte ptr [ebp-4],al 第一種在讀取時直接就把字元串中的元素讀到寄存器cl中,而第二種則要先把指針值讀到edx中,在根據edx讀取字元,顯然慢了。
7.小結:
堆和棧的區別可以用如下的比喻來看出: 使用棧就象我們去飯館里吃飯,只管點菜(發出申請)、付錢、和吃(使用),吃飽了就走,不必理會切菜、洗菜等准備工作和洗碗、刷鍋等掃尾工作,他的好處是快捷,但是自由度小。 使用堆就象是自己動手做喜歡吃的菜餚,比較麻煩,但是比較符合自己的口味,而且自由度大。
[編輯本段]堆和棧的主要分別:
操作系統方面的堆和棧,如上面說的那些,不多說了。 還有就是數據結構方面的堆和棧,這些都是不同的概念。這里的堆實際上指的就是(滿足堆性質的)優先隊列的一種數據結構,第1個元素有最高的優先權;棧實際上就是滿足後進先出的性質的數學或數據結構。 雖然堆棧,堆棧的說法是連起來叫,但是他們還是有很大區別的,連著叫只是由於歷史的原因。堆與棧的分布
[編輯本段]補充
堆棧是一種存儲部件,即數據的寫入跟讀出不需要提供地址,而是根據寫入的順序決定讀出的順序

『陸』 java 堆棧 裡面存的都是些什麼

棧裡面存放到主要是基本數據類型的局部變數和對象的引用(指向對象一種類似地址的東西)。
堆內存主要存放new出來的對象(包括數組)。
其實Java的內存結構不光包括棧和堆,還包括代碼區(載入class類文件本身)、數據區之類的。

『柒』 什麼是堆

堆通常是一個可以被看做一棵樹的數組對象。堆總是滿足下列性質:

1 堆中某個節點的值總是不大於或不小於其父節點的值;

2 堆總是一棵完全二叉樹。

將根節點最大的堆叫做最大堆或大根堆,根節點最小的堆叫做最小堆或小根堆。常見的堆有二叉堆、斐波那契堆等。

(7)堆存儲的內容擴展閱讀

堆的實現通過構造二叉堆(binary heap),實為二叉樹的一種;由於其應用的普遍性,當不加限定時,均指該數據結構的這種實現。這種數據結構具有以下性質。任意節點小於(或大於)它的所有後裔,最小元(或最大元)在堆的根上(堆序性)。

堆總是一棵完全樹。即除了最底層,其他層的節點都被元素填滿,且最底層盡可能地從左到右填入。

堆棧的基本特點:先入後出,後入先出。除頭尾節點之外。

『捌』 淺析棧區和堆區內存分配的區別

堆和棧的區別(內存和數據結構)

在計算機領域,堆棧是一個不容忽視的概念,我們編寫的C語言程序基本上都要用到。但對於很多的初學著來說,堆棧是一個很模糊的概念。堆棧:一種數據結構、一個在程序運行時用於存放的地方,這可能是很多初學者的認識,因為我曾經就是這么想的和匯編語言中的堆棧一詞混為一談。我身邊的一些編程的朋友以及在網上看帖遇到的朋友中有好多也說不清堆棧,所以我想有必要給大家分享一下我對堆棧的看法,有說的不對的地方請朋友們不吝賜教,這對於大家學習會有很大幫助。
數據結構的棧和堆
首先在數據結構上要知道堆棧,盡管我們這么稱呼它,但實際上堆棧是兩種數據結構:堆和棧。

堆和棧都是一種數據項按序排列的數據結構。

棧就像裝數據的桶或箱子
我們先從大家比較熟悉的棧說起吧,它是一種具有後進先出性質的數據結構,也就是說後存放的先取,先存放的後取。這就如同我們要取出放在箱子裡面底下的東西(放入的比較早的物體),我們首先要移開壓在它上面的物體(放入的比較晚的物體)。

堆像一棵倒過來的樹
而堆就不同了,堆是一種經過排序的樹形數據結構,每個結點都有一個值。通常我們所說的堆的數據結構,是指二叉堆。堆的特點是根結點的值最小(或最大),且根結點的兩個子樹也是一個堆。由於堆的這個特性,常用來實現優先隊列,堆的存取是隨意,這就如同我們在圖書館的書架上取書,雖然書的擺放是有順序的,但是我們想取任意一本時不必像棧一樣,先取出前面所有的書,書架這種機制不同於箱子,我們可以直接取出我們想要的書。

內存分配中的棧和堆
然而我要說的重點並不在這,我要說的堆和棧並不是數據結構的堆和棧,之所以要說數據結構的堆和棧是為了和後面我要說的堆區和棧區區別開來,請大家一定要注意。

下面就說說C語言程序內存分配中的堆和棧,這里有必要把內存分配也提一下,大家不要嫌我啰嗦,一般情況下程序存放在Rom或Flash中,運行時需要拷到內存中執行,內存會分別存儲不同的信息,如下圖所示:

內存中的棧區處於相對較高的地址以地址的增長方向為上的話,棧地址是向下增長的。

棧中分配局部變數空間,堆區是向上增長的用於分配程序員申請的內存空間。另外還有靜態區是分配靜態變數,全局變數空間的;只讀區是分配常量和程序代碼空間的;以及其他一些分區。

來看一個網上很流行的經典例子:

main.cpp
int a = 0; 全局初始化區
char *p1; 全局未初始化區
main()
{
int b; 棧
char s[] = "abc"; 棧
char *p2; 棧
char *p3 = "123456"; 123456\0在常量區,p3在棧上。
static int c =0; 全局(靜態)初始化區
p1 = (char *)malloc(10); 堆
p2 = (char *)malloc(20); 堆
}

0.申請方式和回收方式不同
不知道你是否有點明白了,堆和棧的第一個區別就是申請方式不同:棧(英文名稱是stack)是系統自動分配空間的,例如我們定義一個 char a;系統會自動在棧上為其開辟空間。而堆(英文名稱是heap)則是程序員根據需要自己申請的空間,例如malloc(10);開辟十個位元組的空間。由於棧上的空間是自動分配自動回收的,所以棧上的數據的生存周期只是在函數的運行過程中,運行後就釋放掉,不可以再訪問。而堆上的數據只要程序員不釋放空間,就一直可以訪問到,不過缺點是一旦忘記釋放會造成內存泄露。還有其他的一些區別我認為網上的朋友總結的不錯這里轉述一下:

1.申請後系統的響應

棧:只要棧的剩餘空間大於所申請空間,系統將為程序提供內存,否則將報異常提示棧溢出。

堆:首先應該知道操作系統有一個記錄空閑內存地址的鏈表,當系統收到程序的申請時,會遍歷該鏈表,尋找第一個空間大於所申請空間的堆。

結點,然後將該結點從空閑結點鏈表中刪除,並將該結點的空間分配給程序,另外,對於大多數系統,會在這塊內存空間中的首地址處記錄本次分配的大小,這樣,代碼中的 delete語句才能正確的釋放本內存空間。另外,由於找到的堆結點的大小不一定正好等於申請的大小,系統會自動的將多餘的那部分重新放入空閑鏈表中。
也就是說堆會在申請後還要做一些後續的工作這就會引出申請效率的問題。

2.申請效率的比較
根據第0點和第1點可知。

棧:由系統自動分配,速度較快。但程序員是無法控制的。

堆:是由new分配的內存,一般速度比較慢,而且容易產生內存碎片,不過用起來最方便。

3.申請大小的限制棧:在Windows下,棧是向低地址擴展的數據結構,是一塊連續的內存的區域。這句話的意思是棧頂的地址和棧的最大容量是系統預先規定好的,在 WINDOWS下,棧的大小是2M(也有的說是1M,總之是一個編譯時就確定的常數),如果申請的空間超過棧的剩餘空間時,將提示overflow。因此,能從棧獲得的空間較小。
堆:堆是向高地址擴展的數據結構,是不連續的內存區域。這是由於系統是用鏈表來存儲的空閑內存地址的,自然是不連續的,而鏈表的遍歷方向是由低地址向高地址。堆的大小受限於計算機系統中有效的虛擬內存。由此可見,堆獲得的空間比較靈活,也比較大。

4.堆和棧中的存儲內容
由於棧的大小有限,所以用子函數還是有物理意義的,而不僅僅是邏輯意義。

棧: 在函數調用時,第一個進棧的是主函數中函數調用後的下一條指令(函數調用語句的下一條可執行語句)的地址,然後是函數的各個參數,在大多數的C編譯器中,參數是由右往左入棧的,然後是函數中的局部變數。注意靜態變數是不入棧的。
當本次函數調用結束後,局部變數先出棧,然後是參數,最後棧頂指針指向最開始存的地址,也就是主函數中的下一條指令,程序由該點繼續運行。
堆:一般是在堆的頭部用一個位元組存放堆的大小。堆中的具體內容有程序員安排。

關於存儲內容還可以參考這道題。這道題還涉及到局部變數的存活期。

5.存取效率的比較char s1[] = "aaaaaaaaaaaaaaa";
char *s2 = "bbbbbbbbbbbbbbbbb";
aaaaaaaaaaa是在運行時刻賦值的;放在棧中。
而bbbbbbbbbbb是在編譯時就確定的;放在堆中。
但是,在以後的存取中,在棧上的數組比指針所指向的字元串(例如堆)快。
比如:
#include
void main()
{
char a = 1;
char c[] = "1234567890";
char *p ="1234567890";
a = c[1];
a = p[1];
return;
}
對應的匯編代碼
10: a = c[1];
00401067 8A 4D F1 mov cl,byte ptr [ebp-0Fh]
0040106A 88 4D FC mov byte ptr [ebp-4],cl
11: a = p[1];
0040106D 8B 55 EC mov edx,dword ptr [ebp-14h]
00401070 8A 42 01 mov al,byte ptr [edx+1]
00401073 88 45 FC mov byte ptr [ebp-4],al

關於堆和棧區別的比喻
堆和棧的區別可以引用一位前輩的比喻來看出:
使用棧就象我們去飯館里吃飯,只管點菜(發出申請)、付錢、和吃(使用),吃飽了就走,不必理會切菜、洗菜等准備工作和洗碗、刷鍋等掃尾工作,他的好處是快捷,但是自由度小。

使用堆就象是自己動手做喜歡吃的菜餚,比較麻煩,但是比較符合自己的口味,而且自由度大。比喻很形象,說的很通俗易懂,不知道你是否有點收獲。

『玖』 堆內存和棧內存的區別

1、內存區域不同

堆內存是區別於棧區、全局數據區和代碼區的另一個內存區域。堆允許程序在運行時動態地申請某個大小的內存空間。棧內存在函數中定義的一些基本類型的變數和對象的引用變數都在函數的棧內存中分配。

2、特點不同

堆內存實際上指的就是優先隊列的一種數據結構,第1個元素有最高的優先權;棧內存實際上就是滿足先進後出的性質的數學或數據結構。棧內存是存取速度比堆要快,僅次於寄存器,棧數據可以共享。

3、范圍不同

堆內存中分配的內存需要程序員手動釋放,如果不釋放,而系統內存管理器又不自動回收這些堆內存的話動態分配堆內存,那就一直被佔用。棧內存中為這個變數分配內存空間,當超過變數的作用域後,Java會自動釋放掉為該變數所分配的內存空間,該內存空間可以立即被另作他用。