老師,你肯定Java註釋不會被執行嗎?

以前在博客上分享過一篇文章,涉及到 Java 中的註釋,就信誓旦旦地寫了一句話:「註釋是不會被執行的!」結果,有小夥伴留言說,「老師,你肯定嗎?」javascript

我這我的一直有個優勢,就是能聽得進去別人的聲音,管你是讚美的仍是批評的,歷來都是虛心接受。由於我相信,大多數小夥伴都是出於善的目的。java

何況,我在技術上歷來沒想過要成爲多牛逼的大佬,就是喜歡分享的感受,而已。不少文章中出現的錯誤,我都原封不動的保留,由於若是把修正了,那麼留言中那些指出錯誤的人,在後來的讀者眼裏,就會以爲不合時宜。nginx

那些 diss 個人小夥伴們,放心,我是不會介意的。git

儘管如此,但對於註釋這件事,真的是不能忍啊!註釋確定不會被執行啊,我想這位小夥伴必定是在諷刺我。因而我就私信問他爲何,而後他就甩給了我下面這段代碼:程序員

public class Test {
    public static void main(String[] args) {
        String name = "沉默王二";
        // \u000dname="沉默王三";
        System.out.println(name);
    }
}
複製代碼

我拷貝到 IDEA 中跑了一下,結果程序輸出的結果出乎個人意料:github

沉默王三
複製代碼

居然是王三,不是王二。看到這個結果,我算是完全懵逼了。web

那一剎那,我感受這十來年的 Java 算是白學了。大學那會,老師說註釋是不會執行的;就連《編程思想》裏也說註釋是不會執行的。那如今誰能告訴我這到底爲何?面試

不是說程序的世界很單純嗎?不是 0 就是 1?事情搞到這個地步,只能花心思好好研究一下了。編程

單純從代碼上來看,問題應該出在那串特殊的字符上——\u000d,若是不是它在做怪,把 name 的值由「沉默王二」修改成了「沉默王三」,就沒有別的緣由了——沒別的,憑藉多年的工做經驗,找問題的根源我仍是很駕輕就熟的。微信

\u000d 雖然看上去比較陌生,但我知道它是一個 Unicode 字符。問了一下搜索引擎後,知道它表明一個換行符——一種恍然大悟的感受啊。我知道,Java 編譯器不只會編譯代碼,還會解析 Unicode 字符。

我大體看了一眼上面這段代碼編譯後的字節碼,它長下面這個樣子:

// class version 58.65535 (-65478)
// access flags 0x21
public class com/cmower/dzone/secret/Test {

  // compiled from: Test.java

  // access flags 0x1
  public <init>()V
   L0
    LINENUMBER 3 L0
    ALOAD 0
    INVOKESPECIAL java/lang/Object.<init> ()V
    RETURN
   L1
    LOCALVARIABLE this Lcom/cmower/dzone/secret/Test; L0 L1 0
    MAXSTACK = 1
    MAXLOCALS = 1

  // access flags 0x9
  public static main([Ljava/lang/String;)V
   L0
    LINENUMBER 5 L0
    LDC "\u6c89\u9ed8\u738b\u4e8c"
    ASTORE 1
   L1
    LINENUMBER 6 L1
    LDC "\u6c89\u9ed8\u738b\u4e09"
    ASTORE 1
   L2
    LINENUMBER 7 L2
    GETSTATIC java/lang/System.out : Ljava/io/PrintStream;
    ALOAD 1
    INVOKEVIRTUAL java/io/PrintStream.println (Ljava/lang/String;)V
   L3
    LINENUMBER 8 L3
    RETURN
   L4
    LOCALVARIABLE args [Ljava/lang/String; L0 L4 0
    LOCALVARIABLE name Ljava/lang/String; L1 L4 1
    MAXSTACK = 2
    MAXLOCALS = 2
}
複製代碼

嗯,表示看不懂。不過不要緊,把它反編譯一下就好了,因而我看到下面這段代碼:

public class Test {
    public Test() {
    }

    public static void main(String[] args) {
        String name = "沉默王二";
        name = "沉默王三";
        System.out.println(name);
    }
}
複製代碼

咦,兩個反斜槓 // 真的不見了,這能夠肯定一點——註釋確實是不會執行的。只不過 \u000dname="沉默王三"; 擠到了 // 註釋的下一行,就好像下面這段代碼的樣子:

public class Test {
    public static void main(String[] args) {
        String name = "沉默王二";
        //
        name="沉默王三";
        System.out.println(name);
    }
}
複製代碼

那這算不算是 Java 的 bug 呢?說算也不算。

由於經過容許 Java 源代碼包含 Unicode 字符,能夠確保在世界上任何一個區域編寫的代碼在其餘地方執行。

老實說,這段話是我從網上找到,好像明白點啥,又好像不明白。那再來看一段代碼:

double π = Math.PI;
System.out.println(\u03C0);
複製代碼

假如說程序員小王在建立週期率這個變量的時候,不知道 π 這個字符怎麼敲出來,那麼他就能夠選擇使用 \u03C0 來替代——編譯器知道 \u03C0 就是 π 這個變量(編譯器會在編譯其餘代碼以前先解析 Unicode 字符)。

只能說 \u000d 是一種例外吧。

固然了,除非特殊狀況,不要在源代碼中包含 Unicode 字符,以避免更改源代碼的本意。

這篇文章沒有別的意思,我也不想探究過於深奧的東西,純粹是提升一下小夥伴們的認知:註釋有可能被編譯器執行。就好像,魯迅若是不知道茴香豆的「茴」字有 4 種寫法,那他就沒辦法讓孔乙己在魯鎮的那家茶館裏裝逼。

固然了,若是有小夥伴想體驗一下裝逼的感受的話,能夠把下面這段代碼保存在一個名叫 Ugly.java 的文件中:

\u0070\u0075\u0062\u006c\u0069\u0063\u0020\u0020\u0020\u0020

\u0063\u006c\u0061\u0073\u0073\u0020\u0055\u0067\u006c\u0079

\u007b\u0070\u0075\u0062\u006c\u0069\u0063\u0020\u0020\u0020

\u0020\u0020\u0020\u0020\u0073\u0074\u0061\u0074\u0069\u0063

\u0076\u006f\u0069\u0064\u0020\u006d\u0061\u0069\u006e\u0028

\u0053\u0074\u0072\u0069\u006e\u0067\u005b\u005d\u0020\u0020

\u0020\u0020\u0020\u0020\u0061\u0072\u0067\u0073\u0029\u007b

\u0053\u0079\u0073\u0074\u0065\u006d\u002e\u006f\u0075\u0074

\u002e\u0070\u0072\u0069\u006e\u0074\u006c\u006e\u0028\u0020

\u0022\u0048\u0065\u006c\u006c\u006f\u0020\u0077\u0022\u002b

\u0022\u006f\u0072\u006c\u0064\u0022\u0029\u003b\u007d\u007d
複製代碼

在命令行中先執行 javac Ugly.java,再執行 java Ugly 命令就能夠看到程序結果了:

Hello world
複製代碼

體驗事後,就拉到吧。反正寫這樣的代碼誰也看不懂,除了機器。好了,我親愛的讀者朋友,以上就是本文的所有內容了。是否是感受認知邊界又拓寬了?

我是沉默王二,一枚有趣的程序員。若是以爲文章對你有點幫助,請微信搜索「 沉默王二 」第一時間閱讀,回覆【666】更有我爲你精心準備的 500G 高清教學視頻(已分門別類)。

本文 GitHub 已經收錄,有大廠面試完整考點,歡迎 Star。

原創不易,莫要白票,請你爲本文點個贊吧,這將是我寫做更多優質文章的最強動力。

相關文章
相關標籤/搜索