【Simple Java】深入理解“HelloWorld”小程式

来源:http://www.cnblogs.com/chenpi/archive/2016/05/17/5500453.html
-Advertisement-
Play Games

對於每個Java程式員來說,HelloWorld是一個再熟悉不過的程式。它很簡單,但是這段簡單的代碼能指引我們去深入理解一些複雜的概念。這篇文章,我將探索我們能從這段簡單的代碼中學到什麼。如果你對HelloWorld有獨到的理解,請留下你的評論。 HelloWorld.java 為什麼所有東西都是從 ...


對於每個Java程式員來說,HelloWorld是一個再熟悉不過的程式。它很簡單,但是這段簡單的代碼能指引我們去深入理解一些複雜的概念。這篇文章,我將探索我們能從這段簡單的代碼中學到什麼。如果你對HelloWorld有獨到的理解,請留下你的評論。

HelloWorld.java

public class HelloWorld {
    /**
     * @param args
     */
    public static void main(String[] args) {

        System.out.println("Hello World");
    }
}

為什麼所有東西都是從類開始的

Java程式是基於類構建的,每一個方法,欄位必須存在於類裡面。這是因為Java是面向對象的:一切都是對象,即一個類的實例。相對於函數式編程,面向對象編程有很多優勢,如更加模塊化,可擴展性更好等。

為什麼總是需要有一個“main”方法

main方法是靜態方法,程式的入口;靜態方法意味著這個方法是屬於類,而不是對象。

那為什麼是這樣呢?為什麼不使用非靜態方法作為程式的入口呢?

如果這個方法是非靜態的,那麼在使用這個方法之前需要先創建對象,因為非靜態方法需要由對象來調用。作為一個程式的入口,這樣的設計是不現實的。在沒有雞的情況下,我們不能獲取雞蛋。因此,程式入口被設置為靜態方法。

另外,main方法的入參"String[] args"表明一個字元串數組可以傳入該方法用於執行程式的初始化工作。

HelloWorld的位元組碼

為了運行這個程式,Java文件首先被編譯成位元組碼存入一個.class文件。那麼這個位元組碼文件是怎樣的呢?位元組碼本身是不易讀的,我們使用十六進位編輯器打開它,結果如下:

從上面的位元組碼,我們看到了很多操作碼(如CA, 4C,),它們中的每一個都對應著一個助記碼(如下麵例子中的aload_0),操作碼是不易讀的,但是我們可以使用javap去查看.class文件的助記符形式。

"javap -c"可以列印類中每個方法的反彙編代碼,反彙編代碼即一些指令,這些指定組成了java的位元組碼。

javap -classpath . -c HelloWorld

public class HelloWorld extends java.lang.Object{
public HelloWorld();
  Code:
    0: aload_0
    1: invokespecial #1; //Method java/lang/Object."<init>":()V
    4: return
public static void main(java.lang.String[]);
  Code:
    0: getstatic #2; //Field java/lang/System.out:Ljava/io/PrintStream;
    3: ldc #3; //String Hello World
    5: invokevirtual #4; //Method
        java/io/PrintStream.println:(Ljava/lang/String;)V
    8: return
}

以上代碼包含了兩個方法,一個是構造方法,由編譯器自動插入;另一個是main方法。

在每個方法的下麵,都有一系列的指令,如aload_0,invokespecial #1等。每個指定對應的含義可以查看Java位元組碼指令列表。舉個例子,aload_0載入棧中局部變數的引用,getstatic獲取類中的靜態欄位值。註意getstatic後面的"#2",其指向運行時常量池,常量池是Java運行時數據區域。因此,使用"javap -verbose"命令,可以幫助我們查看常量池。

另外,每條指令的前面都有一個數字,如0,1,4等。在位元組碼文件中,每一個方法都有對應的位元組碼數組。這些數字對應的正是數組的索引,這些數組存放了操作碼和對應參數。每個操作碼長度為一個位元組,可以有0或多個參數,這就是為什麼這些數字不是連續的。

現在,我們可以使用"javap -verbose"命令深入看下這個類:

javap -classpath . -verbose HelloWorld

public class HelloWorld extends java.lang.Object
SourceFile: "HelloWorld.java"
minor version: 0
major version: 50
Constant pool:
const #1 = Method #6.#15; // java/lang/Object."<init>":()V
const #2 = Field #16.#17; // java/lang/System.out:Ljava/io/PrintStream;
const #3 = String #18; // Hello World
const #4 = Method #19.#20; //
java/io/PrintStream.println:(Ljava/lang/String;)V
const #5 = class #21; // HelloWorld
const #6 = class #22; // java/lang/Object
const #7 = Asciz <init>;
const #8 = Asciz ()V;
const #9 = Asciz Code;
const #10 = Asciz LineNumberTable;
const #11 = Asciz main;
const #12 = Asciz ([Ljava/lang/String;)V;
const #13 = Asciz SourceFile;
const #14 = Asciz HelloWorld.java;
const #15 = NameAndType #7:#8;// "<init>":()V
const #16 = class #23; // java/lang/System
const #17 = NameAndType #24:#25;// out:Ljava/io/PrintStream;
const #18 = Asciz Hello World;
const #19 = class #26; // java/io/PrintStream
const #20 = NameAndType #27:#28;// println:(Ljava/lang/String;)V
const #21 = Asciz HelloWorld;
const #22 = Asciz java/lang/Object;
const #23 = Asciz java/lang/System;
const #24 = Asciz out;
const #25 = Asciz Ljava/io/PrintStream;;
const #26 = Asciz java/io/PrintStream;
const #27 = Asciz println;
const #28 = Asciz (Ljava/lang/String;)V;
{
public HelloWorld();
Code:
Stack=1, Locals=1, Args_size=1
0: aload_0
1: invokespecial #1; //Method java/lang/Object."<init>":()V
4: return
LineNumberTable:
line 2: 0
public static void main(java.lang.String[]);
Code:
Stack=2, Locals=1, Args_size=1
0: getstatic #2; //Field java/lang/System.out:Ljava/io/PrintStream;
3: ldc #3; //String Hello World
5: invokevirtual #4; //Method
java/io/PrintStream.println:(Ljava/lang/String;)V
8: return
LineNumberTable:
line 9: 0
line 10: 8
}

JVM規範中是這樣描述的:

運行時常量池是為方法服務的,類似於常規程式設計語言中的符號表,但是常量池包含的數據比典型的符號表範圍較廣

"invokespecial #1"指令中的"#1"指向常量池中的#1常量,即"Method #6.#15;",根據這些數字,我們可以遞歸的得到最終常量。行號表可以方便調試人員知道Java源代碼中的哪些行對應位元組碼中的哪些指令。如,Java源代碼中的第9行對應main方法中的code 0,第10行對應code 8。

如果你想要知道更多關於位元組碼的內容,可以嘗試創建一個更加複雜的類,並編譯查看。相對而言,HelloWorld太簡單了。

HelloWorld在JVM中是如何運行的

現在的問題是JVM如何裝載Java類以及如何調用main方法?

在main方法執行之前,JVM需要完成以下步驟,

  • 裝載:裝載類或介面的位元組碼到JVM中
  • 鏈接:將Java類的二進位代碼合併到JVM的運行狀態之中的進程,包含3個步驟(驗證:確保類或介面結構正確、準備:涉及記憶體分配相關、解析:解決符號引用)
  • 初始化:為類的變數初始化合適的值;

載入步驟是由Java類載入器完成的,在JVM啟動的時候,使用了三個類載入器:

  • 引導類載入器:載入/jre/lib下的Java核心類庫,這些類是Java的核心,使用本地代碼編寫。
  • 擴展類載入器:載入擴展目錄下的代碼(如/jar/lib/ext目錄)
  • 系統類載入器:載入CLASSPATH下的代碼

所以HelloWorld是由系統類載入器載入的,當main方法執行之前,會觸發載入,鏈接,初始化其它依賴類操作。

最終,main方法被push到JVM棧中,程式計數器開始做相應操作,將println方法幀push到JVM棧中,當main方法執行完畢,棧中對應的數據被彈出,然後執行完畢。

 

譯文鏈接:http://www.programcreek.com/2013/04/what-can-you-learn-from-a-java-helloworld-program/

 


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

-Advertisement-
Play Games
更多相關文章
  • 通用類型的SWAP函數的實現 通用數組類型的線性搜索函數的實現 c void lSerch(void key,void base,int n,int elemSize){ void elemAddr; for(int i;i ...
  • Scrapy是一個為了爬取網站數據,提取結構性數據而編寫的應用框架。 其可以應用在數據挖掘,信息處理或存儲歷史數據等一系列的程式中。其最初是為了頁面抓取 (更確切來說, 網路抓取 )所設計的, 也可以應用在獲取API所返回的數據(例如 Amazon Associates Web Services ) ...
  • 代碼結構: 1、ThymeleafController 1 package com.xxx.firstboot.web; 2 3 import org.springframework.stereotype.Controller; 4 import org.springframework.ui.Mod ...
  • Implement strStr(). Returns the index of the first occurrence of needle in haystack, or -1 if needle is not part of haystack. 暴力 KMP 以後寫。。 ...
  • 對於三目運算符(ternary operator),python可以用conditional expressions來替代 如對於x<5?1:0可以用下麵的方式來實現 1if x<5else 0 註: conditional expressions是在python 2.5之前引入的,所以以上代碼僅適 ...
  • 思維導圖 點擊下圖,可以看具體內容! 介紹 正則表達式,大家在開發中應該是經常用到,現在很多開發語言都有正則表達式的應用,比如javascript,java,.net,php等等,我今天就把我對正則表達式的理解跟大家嘮嘮,不當之處,請多多指教! 需要知道的術語——下麵的術語你知道多少? 需要知道的術 ...
  • 程式計數器 程式計數器是一塊較小的記憶體,它是線程私有的,可以看作是當前線程執行位元組碼的計數器。在虛擬機的概念模型中,位元組碼解釋器就是通過這個計數器來找到下一個將要執行的指令。java中分支語句,迴圈,異常處理以及線程恢復都是通過程式計數器來實現的。 由於JVM在執行線程的時候是通過CPU輪流執行各個 ...
  • 0. XXXX "is not a class or namespace"錯誤 最詭異的錯誤,提示意思很明顯,說你寫的名字既不是一個類也不是一個命名空間,雖然我C++水平不是很高,但再愚笨也不至於連類的格式class MyClass{....};也寫不明白吧,報此錯誤原因顯然跟它沒關係,那又是怎麼回 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...