Java從內(nèi)存角度帶你理解數(shù)組名實質(zhì)是個地址的論述
本文從Java語言的角度,探討一維數(shù)組與二維數(shù)組的內(nèi)存解析。
一、內(nèi)存的簡化結(jié)構(gòu)
下圖即為內(nèi)存的簡化結(jié)構(gòu)。在Java語言中,內(nèi)存的存儲分配是這樣的:
棧:局部變量
堆:new出來的東西,如對象、數(shù)組等
方法區(qū):包括靜態(tài)域(static)和常量池(String的內(nèi)容就存儲在這里)

內(nèi)存的簡化結(jié)構(gòu)
這張內(nèi)存簡化圖非常重要,需要大家留有印象。
接下來我們在該圖和結(jié)論的基礎(chǔ)上,分步來看一維數(shù)組與二維數(shù)組的內(nèi)存解析。
二、一維數(shù)組的內(nèi)存解析
1. 分步解析
示例代碼
public class Test{
public static void main(String args[]){
int[] arr;
arr = new int[10];
for ( int i=0; i<10; i++ ) {
arr[i] =2*i+1;
System.out.println(arr[i]);
}
}
}step1int[] arr;
此時在棧中創(chuàng)建出了變量arr:

step2 arr= new int[10];
接著使用new關(guān)鍵字,來創(chuàng)建一個一維數(shù)組。需要注意的是,基本數(shù)據(jù)類型數(shù)組在顯式賦值之前, Java會自動給它們賦默認值。由于一維數(shù)組的每個元素都是int類型,因而默認值為0.

step3 arr[i] =2*i+1;
在for循環(huán)中遍歷數(shù)組arr并為其元素賦值:

綜合起來看,內(nèi)存狀態(tài)如下:
2. 綜合解析
如圖,創(chuàng)建數(shù)組并賦值的過程可以簡化成如下示意圖。

左側(cè)為棧區(qū),右側(cè)為堆區(qū)
當(dāng)聲明數(shù)組 int[ ] arr時,arr屬于局部變量,在棧中創(chuàng)建。如上圖中,int[ ] arr1和String[ ] arr2的操作執(zhí)行后,在左側(cè)的棧中創(chuàng)建了變量arr1與arr2.若沒有new的操作,實際上還未給數(shù)組開辟存儲空間。
只有當(dāng)通過new關(guān)鍵字創(chuàng)建數(shù)組對象后,系統(tǒng)才在堆中劃分相應(yīng)的存儲空間,并依據(jù)數(shù)組元素的數(shù)據(jù)類型給新劃分的空間自動賦初值。如上圖中,在new int[4]后,堆區(qū)開辟了4個連續(xù)的存儲空間,并賦值為0,而new String[3]后,堆區(qū)又開辟了3個存儲空間用來存儲String類型的數(shù)據(jù),引用類型String默認初值為null.
同時,數(shù)組的地址(即數(shù)組首元素的地址,假設(shè)為0x12ab)賦值給棧區(qū)中的變量arr1,即變量arr1中存著數(shù)組的地址,通過該地址,arr1可以輕松地在堆中找到它對應(yīng)的數(shù)組。
我們通過中括號 [ ] 來訪問數(shù)組中的各個元素。我們執(zhí)行arr1[0] = 10;這一代碼時,實際便是根據(jù)棧內(nèi)arr1中存著的地址,找到將堆區(qū)中的數(shù)組空間,并將第一個空間中的0改為了10.
而對于arr2[ ]數(shù)組,在new過之后又new了一次,第二次通過new String[5]開辟了一片5個存儲空間的數(shù)組。這時變量arr2中原本存著的地址0x34ab被新地址0x78cd覆蓋,arr2變量存放了新數(shù)組的地址。原數(shù)組在后續(xù)的某個時間內(nèi),將被自動回收。
當(dāng)然,當(dāng)棧區(qū)存有數(shù)組地址的變量arr1與變量arr2最終出棧后,在堆區(qū)劃分的數(shù)組空間也將在之后被回收。
三、二(多)維數(shù)組的內(nèi)存解析
1. 綜合解析
二維數(shù)組是“數(shù)組的數(shù)組”,即一個一維數(shù)組中每個元素也是數(shù)組。由于數(shù)組既可以存儲基本數(shù)據(jù)類型,也可以存儲引用數(shù)據(jù)類型,因而“數(shù)組中存數(shù)組”的理解是可行的。
二維數(shù)組的創(chuàng)建過程與一維數(shù)組類似,這里便不再分步解析。我們直接來看內(nèi)存解析圖:

二維數(shù)組的內(nèi)存解析
與一維數(shù)組的不同之處在于,二維數(shù)組中外層元素也用于存儲地址。在創(chuàng)建二維數(shù)組時,除了在堆區(qū)中為外層元素(一維數(shù)組)開辟了存儲空間外,還為內(nèi)層元素開辟了存儲空間。同時,內(nèi)層元素的首元素地址返回給外層元素。
這樣,數(shù)組名arr1與一維數(shù)組名arr1[ ] 像橋梁一樣連接到內(nèi)層元素arr1[ ][ ]。通過地址和存有地址的棧區(qū)變量arr1、堆區(qū)中的一維數(shù)組arr1[ ],我們可以輕松地訪問到內(nèi)層元素。
其中,內(nèi)層元素的長度可以不相等。int[ ][ ] arr = new int[ ][ ]{{1,2,3},{4,5},{6,7,8}};這樣也是可行的,從圖中就能清晰的看出,內(nèi)層元素之間其實是相對獨立的。
2. 默認初始化方式對初始值的影響
針對于 int[][] arr = new int[4][3]; 這樣將內(nèi)外層元素個數(shù)都指定了的初始化方式,外層元素的初始化值為地址值,內(nèi)層元素的初始化值則與一維數(shù)組初始化情況相同(由元素的數(shù)據(jù)類型而決定)。
而針對 int[][] arr = new int[4][]; 這樣省略內(nèi)層元素長度的初始化方式而言,外層元素的初始化值為null(相當(dāng)于未賦值的引用數(shù)據(jù)類型),內(nèi)層元素則根本沒有初始化值而言(空間還沒開辟),不能調(diào)用,否則編譯器將報錯。
測試
//測試代碼
public class ArrayTest {
public static void main(String[] args) {
int[][] arr = new int[4][3];
System.out.println(arr[0]); //[I@15db9742
System.out.println(arr[0][0]); //0
//System.out.println(arr);
System.out.println("***********************");
float[][] arr1 = new float[4][3];
System.out.println(arr1[0]); //地址值
System.out.println(arr1[0][0]); //0.0
System.out.println("***********************");
String[][] arr2 = new String[4][2];
System.out.println(arr2[1]); //地址值
System.out.println(arr2[1][1]); //null
System.out.println("*********************");
double[][] arr3 = new double[4][];
System.out.println(arr3[1]); //null
// System.out.println(arr3[1][0]); //報錯
}
}總結(jié)
單看數(shù)組名,實際上是一個創(chuàng)建在棧區(qū)的局部變量。整個數(shù)組數(shù)據(jù)量可能較大,直接把數(shù)組內(nèi)所有的元素都存放在棧區(qū)是不太妥當(dāng)?shù)?。因而,?shù)組的主體部分實際上開辟在堆區(qū)。
要想訪問數(shù)組,若堆區(qū)的內(nèi)容與棧區(qū)的數(shù)組名arr之間沒有任何關(guān)系,是無法找到想要訪問的數(shù)組內(nèi)容的。因而,堆區(qū)會返回開辟的數(shù)組空間首元素的地址作為數(shù)組主體的地址,傳給棧區(qū)的局部變量arr(它是一個數(shù)組名)。若是二維數(shù)組,內(nèi)層元素的首元素地址會作為外層元素(也就是一維數(shù)組名)的內(nèi)容。
通過地址和存儲地址的“數(shù)組名”、“一維數(shù)組名”,我們可以一連串地找到我們想要訪問的數(shù)組元素。
該部分內(nèi)容我用文字表述可能不夠簡練,大家更多地可以看圖,圖為重點,通過圖示來理解內(nèi)存解析更好一些。
到此這篇關(guān)于Java從內(nèi)存角度帶你理解數(shù)組名實質(zhì)是個地址的論述的文章就介紹到這了,更多相關(guān)Java數(shù)組名內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Java數(shù)組(Array)最全匯總(上篇)
- Java之?dāng)?shù)組在指定位置插入元素實現(xiàn)
- Java自定義一個變長數(shù)組的思路與代碼
- Java中如何將?int[]?數(shù)組轉(zhuǎn)換為?ArrayList(list)
- Java中將 int[] 數(shù)組 轉(zhuǎn)換為 List分享
- java如何將int數(shù)組轉(zhuǎn)化為Integer數(shù)組
- 淺談Java當(dāng)作數(shù)組的幾個應(yīng)用場景
- 計算Java數(shù)組長度函數(shù)的方法以及代碼分析
- Java C++題解leetcode915分割數(shù)組示例
- Java?從json提取數(shù)組并轉(zhuǎn)換為list的操作方法
- Java數(shù)據(jù)結(jié)構(gòu)之稀疏數(shù)組的實現(xiàn)與應(yīng)用
- Java?C++題解leetcode1441用棧操作構(gòu)建數(shù)組示例
- Java postgresql數(shù)組字段類型處理方法詳解
- Java中數(shù)組的常見操作合集
- 關(guān)于Java?SE數(shù)組的深入理解
- Java二維數(shù)組與稀疏數(shù)組相互轉(zhuǎn)換實現(xiàn)詳解
- Java數(shù)組隊列及環(huán)形數(shù)組隊列超詳細講解
- Java數(shù)組(Array)最全匯總(中篇)
相關(guān)文章
Java?嵌入數(shù)據(jù)引擎從?SQLite?到?SPL詳解
這篇文章主要介紹了Java?嵌入數(shù)據(jù)引擎:從?SQLite?到?SPL,SQLite架構(gòu)簡單,其核心雖然是C語言開發(fā)的,但封裝得比較好,對外呈現(xiàn)為一個小巧的Jar包,能方便地集成在Java應(yīng)用中,本文給大家介紹的非常詳細,需要的朋友參考下2022-07-07
SpringBoot+JWT實現(xiàn)單點登錄完美解決方案
單點登錄是一種統(tǒng)一認證和授權(quán)機制,指在多個應(yīng)用系統(tǒng)中,用戶只需要登錄一次就可以訪問所有相互信任的系統(tǒng),不需要重新登錄驗證,這篇文章主要介紹了SpringBoot+JWT實現(xiàn)單點登錄解決方案,需要的朋友可以參考下2023-07-07
spring boot 測試單元修改數(shù)據(jù)庫不成功的解決
這篇文章主要介紹了spring boot 測試單元修改數(shù)據(jù)庫不成功的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-09-09
MyBatis中selectKey標(biāo)簽及主鍵回填實現(xiàn)
<selectKey>標(biāo)簽在MyBatis中提供了一種靈活的方式來生成和回填主鍵,本文就來介紹一下selectKey標(biāo)簽及主鍵回填實現(xiàn),具有一定的參考價值,感興趣的可以了解一下2024-12-12
java如何利用poi解析doc和docx中的數(shù)據(jù)
這篇文章主要給大家介紹了關(guān)于java如何利用poi解析doc和docx中數(shù)據(jù)的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04
SpringCloud OpenFeign與Ribbon客戶端配置詳解
在springcloud中,openfeign是取代了feign作為負載均衡組件的,feign最早是netflix提供的,他是一個輕量級的支持RESTful的http服務(wù)調(diào)用框架,內(nèi)置了ribbon,而ribbon可以提供負載均衡機制,因此feign可以作為一個負載均衡的遠程服務(wù)調(diào)用框架使用2022-11-11
java中Webclient對象如何解析400狀態(tài)碼詳解
這篇文章主要介紹了java中Webclient對象如何解析400狀態(tài)碼的相關(guān)資料,文中通過代碼介紹的非常詳細,對大家學(xué)習(xí)或者使用java具有一定的參考借鑒價值,需要的朋友可以參考下2024-12-12
Springboot開發(fā)之利用Docker和Kubernetes部署微服務(wù)
這篇文章主要介紹了如何將Spring Boot開發(fā)的微服務(wù)通過Docker容器化,并使用Kubernetes進行部署和管理,幫助讀者掌握現(xiàn)代云原生應(yīng)用的完整開發(fā)部署流程,有需要的可以了解下2025-03-03
在SpringBoot中實現(xiàn)斷點續(xù)傳的實例代碼
在 Spring Boot 或任何其他 web 開發(fā)框架中,斷點續(xù)傳是一種技術(shù),允許文件的傳輸在中斷后可以從中斷點重新開始,而不是從頭開始,種技術(shù)在處理大文件或在不穩(wěn)定的網(wǎng)絡(luò)環(huán)境中尤為重要,本文給大家介紹了SpringBoot中實現(xiàn)斷點續(xù)傳的實例代碼,需要的朋友可以參考下2024-07-07

