深入理解Java記憶體(圖解堆棧)

来源:https://www.cnblogs.com/lipeineng/archive/2018/01/26/8358601.html
-Advertisement-
Play Games

深入理解Java記憶體(圖解)--轉載 深入理解Java記憶體(圖解) 這篇文章是轉自http://blog.csdn.net/shimiso/article/details/8595564博文,自己對其中一些東西加入了自己的理解和補充。 進入正題前首先要知道的是Java程式運行在JVM(Java V ...


深入理解Java記憶體(圖解)--轉載

深入理解Java記憶體(圖解)

  • 這篇文章是轉自http://blog.csdn.net/shimiso/article/details/8595564博文,自己對其中一些東西加入了自己的理解和補充。
  • 進入正題前首先要知道的是Java程式運行在JVM(Java Virtual 
    Machine,Java虛擬機)上,可以把JVM理解成Java程式和操作系統之間的橋梁,JVM實現了Java的平臺無關性,由此可見JVM的重要性。所以在學習Java記憶體分配原理的時候一定要牢記這一切都是在JVM中進行的,JVM是記憶體分配原理的基礎與前提。
  • 簡單通俗的講,一個完整的Java程式運行過程會涉及以下記憶體區域:
  • 寄存器:JVM內部虛擬寄存器,存取速度非常快,程式不可控制。
  • 棧:保存局部變數的值:包括1.基本數據類型的值。2.保存類的實例,即堆區對象的引用(指針)。3.保存載入方法時的幀。
  • :用來存放動態產生的數據,比如new出來的對象。註意創建出來的對象只包含屬於各自的成員變數,並不包括成員方法。因為同一個類擁有各自的成員變數,存儲在堆中的不同位置,但是同一個類不同實例的他們共用該類的方法,並不是每創建一個對象就把成員方法複製一次。
  • 常量池:JVM為每個載入的類型維護一個常量池,常量池是這個類型用到的常量的集合。包括直接常量(基本類型,String)和對其他類型、方法、欄位的符號引用(1)。池中的數據和數組一樣通過索引訪問。由於常量池包含了一個類型所有的對其他類型、方法、欄位的符號引用,所以常量池在Java的動態鏈接中起了核心作用。常量池存在於方法區中,而方法區存在於堆中。

下麵是記憶體表示圖: 
            這裡寫圖片描述

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

預備知識:

1.一個Java文件,只要有main入口方法,我們就認為這是一個Java程式,可以單獨編譯運行。 
2.無論是普通類型的變數還是引用類型的變數(俗稱實例),都可以作為局部變數,他們都可以出現在棧中。只不過普通類型的變數在棧中直接保存它所對應的值,而引用類型的變數保存的是一個指向堆區的指針,通過這個指針,就可以找到這個實例在堆區對應的對象。因此,普通類型變數只在棧區占用一塊記憶體,而引用類型變數要在棧區和堆區各占一塊記憶體。 
3.在方法的參數傳遞中,基本數據類型,String類是按值傳遞,即拷貝了一個副本!引用數據類型是按引用傳遞,即把棧中的地址傳入!

示例: 
這裡寫圖片描述
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所占用的棧空間。 
這裡寫圖片描述
調用test對象的change2方法,以實例d1為參數。JVM檢測到change2方法中的b參數為局部變數,立即加入到棧中,由於是引用類型的變數,所以b中保存的是d1中的指針,此時b和d1指向同一個堆中的對象。在b和d1之間傳遞是指針。 
這裡寫圖片描述 
change2方法中又實例化了一個BirthDate對象,並且賦給b。在內部執行過程是:在堆區new了一個對象,並且把該對象的指針保存在棧中的b對應空間,此時實例b不再指向實例d1所指向的對象,但是實例d1所指向的對象並無變化,這樣無法對d1造成任何影響。 
這裡寫圖片描述
change2方法執行完畢,立即釋放局部引用變數b所占的棧空間,註意只是釋放了棧空間,堆空間要等待自動回收。 
這裡寫圖片描述
調用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.類的成員變數在不同對象中各不相同,都有自己的存儲空間(成員變數在堆中的對象中)。而類的方法卻是該類的所有對象共用的,只有一套,對象使用方法的時候方法才被壓入棧,方法不使用則不占用記憶體。 
以上分析只涉及了棧和堆,還有一個非常重要的記憶體區域:常量池,這個地方往往出現一些莫名其妙的問題。常量池是幹嘛的上邊已經說明瞭,也沒必要理解多麼深刻,只要記住它維護了一個已載入類的常量就可以了。接下來結合一些例子說明常量池的特性。

  • 基本類型和基本類型的包裝類。基本類型有:byte、short、char、int、long、boolean。基本類型的包裝類分別是:Byte、Short、Character、Integer、Long、Boolean。註意區分大小寫。二者的區別是:基本類型體現在程式中是普通變數,基本類型的包裝類是類,體現在程式中是引用變數。因此二者在記憶體中的存儲位置不同:基本類型存儲在棧中,而基本類型包裝類存儲在堆中。上邊提到的這些包裝類都實現了常量池技術,另外兩種浮點數類型的包裝類則沒有實現。另外,String類型也實現了常量池技術。
 

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、基本查詢 (表:emp 列:ename ,depton,sal 、表示換行) 1.查當前用戶 SQL> show user 2.查表結構 SQL> desc emp (emp:表名) --查詢出名稱、類型、是否為空 3.清屏 SQL> host cls 4.設置行寬 SQL> show line ...
  • 轉載地址為:http://blog.51cto.com/hades02/1641652 首先在命令行輸入 show global variables like '%general%' ,然後出現下麵的視窗,欄位general_log為開關,一般預設是為OFF的,所以還沒開啟監視,然後general_ ...
  • Redis簡介 Redis是一個開源的使用ANSI C語言編寫、支持網路、可基於記憶體亦可持久化的日誌型、Key-Value資料庫,並提供多種語言的API. Redis是一個開源(BSD許可),記憶體存儲的數據結構伺服器,可用作資料庫,高速緩存和消息隊列代理。它支持字元串、哈希表、列表、集合、有序集合, ...
  • DDL 資料庫定義語言(庫的操作)(看懂,工作時使用較少)1.創建一個庫 create database 庫名稱 [character set 碼表名稱 collate 字元校對集名稱] create database day15; 1>看到Query OK, 1 row affected (0.0 ...
  • 1、數據倉庫.就與我們之前學過的純文本,properties,XML這些技術一樣.用來保存數據.並提供對數據進行增刪改查的操作.我們以後做項目時, 項目中的數據都是保存在資料庫中的. 2、資料庫的特點 2.1、實現數據共用 2.2、減少數據的冗餘度 2.3、數據實現集中控制 2.4、數據一致性,完整 ...
  • 前幾天幫用戶解決了一個問題,記錄下 有個用戶用我們的芝麻小客服,配置都是正確的,但是一直收不到用戶發的消息,無法自動回覆消息。 後來發現原來是有bug,如果發現你登陸小程式的客服系統,然後,別人發送消息,你都看不到,那麼可能是bug,需要向官方反饋。 這裡就有一個用戶反饋的,官方解決了,參考下麵這個 ...
  • 你沒看錯,右上角的那個大圓就是傳說中的太陽,^_^ 這個動畫的難點在於這個“食物”的繪製上吧,不用懷疑,你還是沒看錯,那些小點就是傳說中的食物 首先一步步來,看到這種效果,第一個想到的就是一個普通的小圓,而這個大圓就用貝塞爾繪製,至於為什麼用貝塞爾而不是直接繪製一個半圓呢,因為食物是繞著半圓的,緊貼 ...
  • Java記憶體管理的進一步理解-模擬過程圖解--轉載 java的記憶體管理分為: 1、堆記憶體;2、棧記憶體;3、方法區;4、本地方法區 /* 1:方法區 方法區存放裝載的類數據信息包括: (1):基本信息: 1)每個類的全限定名 2)每個類的直接超類的全限定名(可約束類型轉換) 3)該類是類還是介面 4) ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...