虛擬機 java 開發_深入淺出 Java 虛擬機 · 通往高級 Java 開發的必經之路

第一章 JVM 內存模型

Java 虛擬機(Java Virtual Machine=JVM)的內存空間分為五個部分,分別是:程序計數器

Java 虛擬機棧

本地方法棧

方法區。

下面對這五個區域展開深入的介紹。

1.1 程序計數器

1.1.1 什么是程序計數器?

程序計數器是一塊較小的內存空間,可以把它看作當前線程正在執行的字節碼的行號指示器。也就是說,程序計數器里面記錄的是當前線程正在執行的那一條字節碼指令的地址。

注:但是,如果當前線程正在執行的是一個本地方法,那么此時程序計數器為空。

1.1.2 程序計數器的作用

程序計數器有兩個作用:字節碼解釋器通過改變程序計數器來依次讀取指令,從而實現代碼的流程控制,如:順序執行、選擇、循環、異常處理。

在多線程的情況下,程序計數器用于記錄當前線程執行的位置,從而當線程被切換回來的時候能夠知道該線程上次運行到哪兒了。

1.1.3 程序計數器的特點是一塊較小的存儲空間

線程私有。每條線程都有一個程序計數器。

是唯一一個不會出現OutOfMemoryError的內存區域。

生命周期隨著線程的創建而創建,隨著線程的結束而死亡。

1.2 Java虛擬機棧(JVM Stack)

1.2.1 什么是Java虛擬機棧?

Java虛擬機棧是描述Java方法運行過程的內存模型。

Java虛擬機棧會為每一個即將運行的Java方法創建一塊叫做“棧幀”的區域,這塊區域用于存儲該方法在運行過程中所需要的一些信息,這些信息包括:局部變量表 存放基本數據類型變量、引用類型的變量、returnAddress類型的變量。

操作數棧

動態鏈接

方法出口信息

當一個方法即將被運行時,Java虛擬機棧首先會在Java虛擬機棧中為該方法創建一塊“棧幀”,棧幀中包含局部變量表、操作數棧、動態鏈接、方法出口信息等。當方法在運行過程中需要創建局部變量時,就將局部變量的值存入棧幀的局部變量表中。

當這個方法執行完畢后,這個方法所對應的棧幀將會出棧,并釋放內存空間。

注意:人們常說,Java的內存空間分為“棧”和“堆”,棧中存放局部變量,堆中存放對象。

這句話不完全正確!這里的“堆”可以這么理解,但這里的“棧”只代表了Java虛擬機棧中的局部變量表部分。真正的Java虛擬機棧是由一個個棧幀組成,而每個棧幀中都擁有:局部變量表、操作數棧、動態鏈接、方法出口信息。

1.2.2 Java 虛擬機棧的特點

(1)局部變量表的創建是在方法被執行的時候,隨著棧幀的創建而創建。而且,局部變量表的大小在編譯時期就確定下來了,在創建的時候只需分配事先規定好的大小即可。此外,在方法運行的過程中局部變量表的大小是不會發生改變的。

(2)Java 虛擬機棧會出現兩種異常:StackOverFlowError 和 OutOfMemoryError。a) StackOverFlowError: 若Java虛擬機棧的內存大小不允許動態擴展,那么當線程請求棧的深度超過當前Java虛擬機棧的最大深度的時候,就拋出StackOverFlowError異常。

b) OutOfMemoryError: 若Java虛擬機棧的內存大小允許動態擴展,且當線程請求棧時內存用完了,無法再動態擴展了,此時拋出OutOfMemoryError異常。

(3)Java虛擬機棧也是線程私有的,每個線程都有各自的Java虛擬機棧,而且隨著線程的創建而創建,隨著線程的死亡而死亡。注:StackOverFlowError和OutOfMemoryError的異同?

StackOverFlowError表示當前線程申請的棧超過了事先定好的棧的最大深度,但內存空間可能還有很多。而OutOfMemoryError是指當線程申請棧時發現棧已經滿了,而且內存也全都用光了。

1.3 本地方法棧

1.3.1 什么是本地方法棧?

本地方法棧和Java虛擬機棧實現的功能類似,只不過本地方法區是本地方法運行的內存模型。

本地方法被執行的時候,在本地方法棧也會創建一個棧幀,用于存放該本地方法的局部變量表、操作數棧、動態鏈接、出口信息。

方法執行完畢后相應的棧幀也會出棧并釋放內存空間。

也會拋出StackOverFlowError和OutOfMemoryError異常。

1.4 堆

1.4.1 什么是堆?

堆是用來存放對象的內存空間。 幾乎所有的對象都存儲在堆中。

1.4.2 堆的特點

(1)線程共享

整個 Java 虛擬機只有一個堆,所有的線程都訪問同一個堆。而程序計數器、Java 虛擬機棧、本地方法棧都是一個線程對應一個的。

(2)在虛擬機啟動時創建。

(3)垃圾回收的主要場所。

(4)可以進一步細分為:新生代、老年代。

新生代又可被分為:Eden、From Survior、To Survior。不同的區域存放具有不同生命周期的對象。這樣可以根據不同的區域使用不同的垃圾回收算法,從而更具有針對性,從而更高效。

(5)堆的大小既可以固定也可以擴展,但主流的虛擬機堆的大小是可擴展的,因此當線程請求分配內存,但堆已滿,且內存已滿無法再擴展時,就拋出 OutOfMemoryError。

1.5 方法區

1.5.1 什么是方法區?

Java 虛擬機規范中定義方法區是堆的一個邏輯部分。方法區中存放已經被虛擬機加載的類信息、常量、靜態變量、即時編譯器編譯后的代碼等。

1.5.2 方法區的特點線程共享 方法區是堆的一個邏輯部分,因此和堆一樣,都是線程共享的。整個虛擬機中只有一個方法區。

永久代 方法區中的信息一般需要長期存在,而且它又是堆的邏輯分區,因此用堆的劃分方法,我們把方法區稱為老年代。

內存回收效率低 方法區中的信息一般需要長期存在,回收一遍內存之后可能只有少量信息無效。 對方法區的內存回收的主要目標是:對常量池的回收 和 對類型的卸載。

Java虛擬機規范對方法區的要求比較寬松。 和堆一樣,允許固定大小,也允許可擴展的大小,還允許不實現垃圾回收。

1.5.3 什么是運行時常量池?

方法區中存放三種數據:類信息、常量、靜態變量、即時編譯器編譯后的代碼。其中常量存儲在運行時常量池中。

我們一般在一個類中通過public static final來聲明一個常量。這個類被編譯后便生成Class文件,這個類的所有信息都存儲在這個class文件中。

當這個類被Java虛擬機加載后,class文件中的常量就存放在方法區的運行時常量池中。而且在運行期間,可以向常量池中添加新的常量。如:String類的intern()方法就能在運行期間向常量池中添加字符串常量。

當運行時常量池中的某些常量沒有被對象引用,同時也沒有被變量引用,那么就需要垃圾收集器回收。

1.6 直接內存

直接內存是除Java虛擬機之外的內存,但也有可能被Java使用。

在NIO中引入了一種基于通道和緩沖的IO方式。它可以通過調用本地方法直接分配Java虛擬機之外的內存,然后通過一個存儲在Java堆中的DirectByteBuffer對象直接操作該內存,而無需先將外面內存中的數據復制到堆中再操作,從而提升了數據操作的效率。

直接內存的大小不受Java虛擬機控制,但既然是內存,當內存不足時就會拋出OOM異常。

1.7 綜上所述Java虛擬機的內存模型中一共有兩個“棧”,分別是:Java虛擬機棧和本地方法棧。 兩個“棧”的功能類似,都是方法運行過程的內存模型。并且兩個“棧”內部構造相同,都是線程私有。 只不過Java虛擬機棧描述的是Java方法運行過程的內存模型,而本地方法棧是描述Java本地方法運行過程的內存模型。

Java虛擬機的內存模型中一共有兩個“堆”,一個是原本的堆,一個是方法區。方法區本質上是屬于堆的一個邏輯部分。堆中存放對象,方法區中存放類信息、常量、靜態變量、即時編譯器編譯的代碼。

堆是Java虛擬機中最大的一塊內存區域,也是垃圾收集器主要的工作區域。

程序計數器、Java虛擬機棧、本地方法棧是線程私有的,即每個線程都擁有各自的程序計數器、Java虛擬機棧、本地方法棧。并且他們的生命周期和所屬的線程一樣。 而堆、方法區是線程共享的,在Java虛擬機中只有一個堆、一個方法棧。并在JVM啟動的時候就創建,JVM停止才銷毀。

第二章 揭開Java對象創建的奧秘

2.1 對象的創建過程

當虛擬機遇到一條含有new的指令時,會進行一系列對象創建的操作:

(1)檢查常量池中是否有即將要創建的這個對象所屬的類的符號引用;若常量池中沒有這個類的符號引用,說明這個類還沒有被定義!拋出ClassNotFoundException;

若常量池中有這個類的符號引用,則進行下一步工作;

(2)進而檢查這個符號引用所代表的類是否已經被JVM加載;若該類還沒有被加載,就找該類的class文件,并加載進方法區;

若該類已經被JVM加載,則準備為對象分配內存;

(3)根據方法區中該類的信息確定該類所需的內存大小;

一個對象所需的內存大小是在這個對象所屬類被定義完就能確定的!且一個類所生產的所有對象的內存大小是一樣的!JVM在一個類被加載進方法區的時候就知道該類生產的每一個對象所需要的內存大小。

(4)從堆中劃分一塊對應大小的內存空間給新的對象;分配堆中內存有兩種方式:指針碰撞 如果JVM的垃圾收集器采用復制算法或標記-整理算法,那么堆中空閑內存是完整的區域,并且空閑內存和已使用內存之間由一個指針標記。那么當為一個對象分配內存時,只需移動指針即可。因此,這種在完整空閑區域上通過移動指針來分配內存的方式就叫做“指針碰撞”。

空閑列表 如果JVM的垃圾收集器采用標記-清除算法,那么堆中空閑區域和已使用區域交錯,因此需要用一張“空閑列表”來記錄堆中哪些區域是空閑區域,從而在創建對象的時候根據這張“空閑列表”找到空閑區域,并分配內存。 綜上所述:JVM究竟采用哪種內存分配方法,取決于它使用了何種垃圾收集器。

(5)為對象中的成員變量賦上初始值(默認初始化);

(6)設置對象頭中的信息;

(7)調用對象的構造函數進行初始化;

此時,整個對象的創建過程就完成了。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/370463.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/370463.shtml
英文地址,請注明出處:http://en.pswp.cn/news/370463.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

java.lang.ClassNotFoundException:如何解決

本文適用于當前面臨java.lang.ClassNotFoundException挑戰的Java初學者。 它將為您提供此常見Java異常的概述,這是一個示例Java程序,可支持您的學習過程和解決策略。 如果您對與更高級的類加載器相關的問題感興趣,我建議您復習有關java.lang…

iOS10 App跳轉到系統設置

實現類似萬能鑰匙中點擊一個Wi-Fi跳轉到系統Wi-Fi設置界面的功能。 NSString * urlString "App-Prefs:rootWIFI";if ([[UIApplication sharedApplication] canOpenURL:[NSURL URLWithString:urlString]]) {if ([[UIDevice currentDevice].systemVersion doubleValue…

python生成器 圖片分類_python批量處理圖片圖片Python迭代器和生成器介紹

Python迭代器和生成器介紹迭代器迭代器是一個實現了迭代器協議的對象,Python中的迭代器協議就是有next方法的對象會前進到下一結果,而在一系列結果的末尾是,則會引發StopIteration。在for循環中,Python將自動調用工廠函數iter()獲…

Java-IO流之BufferedReader 和BufferedWriter的使用和原理

BufferedReader和BufferedWriter出現的目的是為了對FileReader以及FileWriter的讀寫操作進行增強,而怎么增強呢,原理類似于使用StringBuilder,是把數據先放入他們的一個char數組中,然后再操作char數組。 使用緩沖區的字符流是使用…

小程序實踐(三):九宮格實現及item跳轉

效果圖: 實現效果圖紅色線包含部分的九宮格效果,并附帶item點擊時間。 ------------------------------------------------------------------------------------------------------ 具體實現: 1、首先添加圖片資源文件 在項目根目錄新建一個…

用JavaFX編寫圖塊引擎

隨著JavaFX嵌入式版本的問世,我們的框架對于游戲開發變得越來越有趣,因為我們現在可以瞄準平板電腦和智能手機等小型消費類設備。 因此,我決定對JavaFX進行更多的游戲編寫實驗。 這次,我想使用Canvas對渲染進行更多控制&#xff0…

python命令行運行模式_[Python] 命令行模式閱讀博客園的博文

1 #-*- coding:UTF-8 -*-2 importrequests3 from lxml importetree4 importsys5 importio6 importos789 sys.stdout io.TextIOWrapper(sys.stdout.buffer, encodinggb18030)101112 classCnBlogs:13 """"14 Auth:reader15 發表地址:…

HTML5--應用網頁模板

因為剛開始寫博客,只想著把知識點記錄在這,也想給你們一些參考,在布局上有些沒有思考太多;回過頭來看,實在是不忍直視,對不住之前閱讀的100 ,既然昨天的事無法挽回,那就從現在開始從新整改吧!也希望大家看了,能對你們有所幫助 1.先給大家看看效果圖,好讓大家有點興趣 2.大家再來…

企業集成模式簡介

在此博客文章中,我們將介紹一些企業集成模式。 這些是旨在解決集成挑戰的已知設計模式。 閱讀此書后,您將可以設計集成解決方案。 EIP(簡而言之)是已知的設計模式,可為應用程序集成過程中遇到的問題/問題提供解決方案…

手把手教你Chrome瀏覽器安裝Postman(含下載云盤鏈接)【轉載】

轉載自:http://www.ljwit.com/archives/php/278.html 說明: Postman不多介紹,是一款功能強大的網頁調試與發送網頁HTTP請求的Chrome插件。本文主要介紹下安裝過程。 本文使用的是解壓文件直接進行安裝。是比較快速有效的安裝方式,…

C語言博客作業--數據類型

題目1:7-4 打印菱形圖案 1. 本題PTA提交列表 2. 設計思路 1.定義變量i,j,k,n;且聲明i為要打印的行數,j是控制輸出打印空格和星星,n是菱形為菱形的高 2.輸入n 3.i1,j1 4.先打印上半部分,第一行到n/21行,輸出…

信息隱藏將txt文件合并到jpg文件中_使用Kali Linux在圖像內隱藏機密消息—可在任何Linux發行版使用

歡迎回到“Esn技術社區”!今天,我們將演示如何使用Steghide(一種可在Kali Linux上使用的流行隱寫工具)在圖像內隱藏消息。在計算機科學中,將信息隱藏在文件內(例如圖像,文檔,程序,有用數據,消息…

Spring 3.1,Cloud Foundry和本地開發

這篇文章將幫助您在Cloud Foundry上使用MongoDB構建Spring 3.1 Web應用程序。 除了推動Cloud Foundry之外,您還可以使用MongoDB實例在本地環境中進行開發。 目標 此博客發布的目標是在本地構建應用程序,然后發布到本地Cloud Foundry實例。 我們將利用C…

Spring MVC 簡述:從MVC框架普遍關注的問題說起

任何一個完備的MVC框架都需要解決Web開發過程中的一些共性的問題,比如請求的收集與分發、數據前后臺流轉與轉換,當前最流行的SpringMVC和Struts2也不例外。本文首先概述MVC模式的分層思想與MVC框架普遍關注的問題,并以此為契機結合SpringMVC的…

java方法調用機制_Java方法調用機制 - osc_bkdv2it5的個人空間 - OSCHINA - 中文開源技術交流社區...

最近在編程時,修改方法傳入對象的對象引用,并沒有將修改反映到調用方法中。奇怪為什么結果沒有變化,原因是遺忘了Java對象引用和內存分配機制。本文介紹3個點:① 該問題舉例說明② 簡要闡述Java內存區域③ 介紹JVM中方法調用的機制…

CSS染色圖標(圖片)

之前一直以為用background引入的圖標無法染色&#xff08;非字體圖標&#xff09;&#xff0c;現在才知道有黑科技可以用&#xff0c;就是利用drop-shadow。 代碼示例 <!DOCTYPE html> <html> <head lang"en"><meta charset"UTF-8"&…

eclipse安裝java web插件

1 查看eclipse版本 找到eclipse的安裝目錄&#xff0c;找到readme文件&#xff0c;打開其中的html文件&#xff0c;我的是4.6版本的,代號是oxygen 2 安裝 打開eclipse,點擊help-Install new software-單擊add&#xff0c;在彈出窗口中輸入網址&#xff1a; http://download.ecl…

python正則表達式指南_Python正則表達式指南

1. 正則表達式基礎1.1. 簡單介紹正則表達式并不是Python的一部分。正則表達式是用于處理字符串的強大工具&#xff0c;擁有自己獨特的語法以及一個獨立的處理引擎&#xff0c;效率上可能不如str自帶的方法&#xff0c;但功能十分強大。得益于這一點&#xff0c;在提供了正則表達…

Google Guava EventBus用于事件編程

在任何軟件應用程序中都是如此&#xff0c;有些對象需要共享信息才能完成工作。 在Java應用程序中&#xff0c;實現信息共享的一種方法是擁有事件偵聽器&#xff0c;其唯一目的是在發生所需事件時采取某些措施。 在大多數情況下&#xff0c;此過程有效&#xff0c;并且最有經驗…

system類

package system.cn; /** system類的方法 都是靜態方法&#xff0c;可以直接用類名直接調用* 常用的方法&#xff1a;* static long currentTimeMillis() 返回以毫秒為單位的當前時間。 static void exit(int status) 終止當前正在運行的 Java 虛擬機。 static void gc…