面試官刁難:Java字元串可以引用傳遞嗎?

来源:https://www.cnblogs.com/qing-gee/archive/2020/01/03/12143003.html
-Advertisement-
Play Games

老讀者都知道了,六年前,我從蘇州回到洛陽,抱著一幅“海歸”的心態,投了不少簡歷,也“約談”了不少面試官,但僅有兩三個令我感到滿意。其中有一位叫老馬,至今還活在我的手機通訊錄里。他當時扔了一個面試題把我砸懵了:“王二,Java 字元串可以引用傳遞嗎?” 我當時二十三歲,正值青春年華,從事 Java 編 ...


老讀者都知道了,六年前,我從蘇州回到洛陽,抱著一幅“海歸”的心態,投了不少簡歷,也“約談”了不少面試官,但僅有兩三個令我感到滿意。其中有一位叫老馬,至今還活在我的手機通訊錄里。他當時扔了一個面試題把我砸懵了:“王二,Java 字元串可以引用傳遞嗎?”

當時二十三歲,正值青春年華,從事 Java 編程已有 N 年經驗(N < 4),自認為所有的面試題都能對答如流,結果沒想到啊,被“刁難”了——原來洛陽這塊互聯網的荒漠也有技術專家啊。現在回想起來,臉上不自覺地泛起了羞愧的紅暈:主要是自己當時太菜了。不管怎麼說,是時候寫篇文章剖析一下字元串是否可以引用傳遞了。

對於絕大多數的初級程式員或者說不重視“內功”的老鳥來說,往往停留在“知其然不知其所以然”的層面上——會用,略知一二,但要求他把問題說清楚的時候,就只能撓撓頭雙手一攤一張問號臉了。

好了,讓我們來步入正題。先來看一段有趣但令人困惑的代碼片段吧。

public static void main(String[] args) {
    String x = new String("沉默王二");
    change(x);
    System.out.println(x);
}

public static void change(String x) {
    x = "沉默王三";
}

從代碼的字面邏輯來看,程式應該輸出“沉默王三”,但事與願違,程式輸出的結果卻是“沉默王二”。change() 方法做的是無用功,因為 String 是值傳遞而不是引用傳遞。引用傳遞可以在被調用的方法中對實參進行修改,但值傳遞卻不可以。為什麼呢?

x 存儲的是一個引用,該引用指向記憶體中的“沉默王二”字元串對象。當我們把 x 作為參數傳遞給 change() 方法時,x 仍然指向的是記憶體中“沉默王二”字元串,就像下麵這幅圖表達的意思一樣。

那麼問題來了。正因為 Java 是值傳遞,x 的值是“沉默王二”的引用。那麼當 change() 方法被調用的時候,x 不是剛好指向了記憶體中新創建的字元串對象“沉默王三”了嗎?就像下麵這幅圖表達的意思那樣。

哦,看起來是一個很完美的解釋,對吧?但這樣的解釋存在一些問題。

當字元串“沉默王二”被創建的時候,Java 會在記憶體中申請一小段空間,用來存儲這個字元串對象。然後呢,把對象的引用指向了變數 x,也就是說,變數 x 實際上存儲的是對象的引用(對象在記憶體中存儲的地址)。

我相信大家對上面這一點(對象和對象引用)已經完全理解了。

關鍵的點來了。當變數 x 作為參數(實參)傳遞給 change() 方法時,實際上傳遞的是 x 的一個拷貝(形參)。在 change() 方法中,形參 x 起先引用的也是“沉默王二”這個對象,當執行 x = "沉默王三" 的時候,會在記憶體中創建新的字元串“沉默王三”,然後形參 x 不再引用“沉默王二”這個對象了,改為引用“沉默王三”這個對象了。但實參 x 呢?並沒有發生任何的改變!就像下麵這幅圖一樣。

假如我們真的需要改變字元串呢?那就不能使用 String 類了,最好使用 StringBuilder,來擼一串代碼吧。

public static void main(String[] args) {
    StringBuilder x = new StringBuilder("沉默王二");
    change(x);
    System.out.println(x);
}

public static void change(StringBuilder x) {
    x.delete(3,4).append("三");
}

上述代碼會輸出“沉默王三”,但假如我們使用 new 關鍵字重新對形參 x 進行賦值,就無濟於事。

public static void main(String[] args) {
    StringBuilder x = new StringBuilder("沉默王二");
    change(x);
    System.out.println(x);
}

public static void change(StringBuilder x) {
    x = new StringBuilder("沉默王三");
}

程式輸出的結果仍然是“沉默王二”,原因其實和 String 一樣,change() 方法在記憶體中創建了新的字元串“沉默王三”,然後形參 x 不再引用“沉默王二”這個對象,改為引用“沉默王三”這個對象了。但實參 x 並沒有任何改變。

看到這,有些讀者可能更疑惑了。x = new StringBuilder("沉默王三") 不可以改變實參,而 x.delete(3,4).append("三") 卻可以,為什麼?為什麼?為什麼?為什麼呢?

不要著急,我們來分析一下 delete() 方法的源碼。

public AbstractStringBuilder delete(int start, int end) {
    int len = end - start;
    if (len > 0) {
        System.arraycopy(value, start+len, value, start, count-end);
        count -= len;
    }
    return this;
}

其中 value 是一個字元數組,用來存儲字元序列;count 用來表示字元序列中實際有效的字元數量。

count -= len 執行之前,value 的字元內容為“沉默王二”,count 為 4。我是怎麼知道的呢?通過 IDEA 的 debug 視圖,截圖為證。

count -= len 執行之後,value 的字元內容仍然為“沉默王二”,但 count 變成了 3。

當滑鼠停留在 this 上時,此時的字元內容為“沉默王”,也就意味著 x 當前的字元內容為“沉默王”。同樣的,當我們在 append() 方法上進行 debug 的時候,也可以觀察到字元串發生變化的細節。

append() 方法執行結束後,此時形參 x 的字元內容為“沉默王三”。

change() 方法執行完後,此時實參 x 的字元內容為“沉默王三”。

通過上面的源碼分析,大家應該會發現另外一個事實:x 對象始終是“StringBuilder@512”,這意味著什麼呢?一圖勝千言,畫個圖大家一看就明白了。

由於形參 x 和實參 x 引用的都是同一個對象,那麼 change() 方法執行結束後,實參 x 的字元內容自然也就發生了變化。

綜上所述:Java 字元串不是引用傳遞而是值傳遞;更進一步的說,Java 只有值傳遞,沒有引用傳遞。

遙想公瑾當年,小喬初嫁了,雄姿英發。

羽扇綸巾,談笑間,檣櫓灰飛煙滅。

故國神游,多情應笑我,早生華髮。

哎,後悔啊,早年我要是能把這道面試題吃透的話,也不用被老馬刁難了。另外,我想要告訴大家的是,作為程式員,我們千萬不要輕視這些基礎的知識點。因為基礎的知識點是各種上層技術共同的基礎,只有徹底地掌握了這些基礎知識點,才能更好地理解程式的運行原理,做出更優化的產品。


好了,各位讀者朋友們,以上就是本文的全部內容了。能看到這裡的都是最優秀的程式員,升職加薪就是你了

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

-Advertisement-
Play Games
更多相關文章
  • 「前端,有發展“錢”途」 前端發展隨著互聯網大時代如火如荼的進行著,Web前端技術依靠其自身在頁面交互效果上強大的功能屬性受到了眾多企業的青睞。這不僅是在北上廣,很多一二線城市都是如此。 無論是你使用的智能手機,還是iPad,還是pc電腦等等,前臺的頁面樣式都需要前端開發工程師來編寫實現,也因此市場 ...
  • //點擊a標簽,輪流顯示和隱藏<div id="timo" style="background-color:red;height:50px;width:50px;"></div><a href="javascript:void(0);" id="lol">點擊這裡</a> <script>$(fun ...
  • 2)局部變數和全局變數 馬克-to-win:瀏覽器裡面 window 就是 global,通常可以省。nodejs 里沒有 window,但是有個叫 global 的。例 3.2.1<html><head> <meta http-equiv="content-type" content="text/ ...
  • What do we need? 筆者目的是在vue項目打包後的 dist/index.html 文件中寫入本次打包git用戶、最後一次git提交信息,這樣做的目的是便於線上項目的管理和防止同事之間的相互扯皮。最後打包出的效果如下圖: How to do? 版本信息需要記錄 git最後一次提交作者( ...
  • javascript當中null和undefined的==和===的比較 ...
  • 本系統包括兩台Web伺服器和個資料庫伺服器,資料庫伺服器採用雙主從配置,另外還有負載均衡以及redis實現session共用 ...
  • ORM一直是長久不衰的話題,各種重覆造輪子的過程一直在進行,輪子都一樣是圓的,你的又有什麼特點呢? CRL這個輪子造了好多年,功能也越來越標準完備,在開發過程中,解決了很多問題,先上一張腦圖描述CRL的功能 開發框架的意義在於 開發更標準,更統一,不會因為不同人寫的代碼不一樣 開發效率更高,無需重新 ...
  • 年前我們一起聊了 程式員為什麼要懂架構、架構是什麼 和 架構都有哪些類型 這三個話題,今天我們來看看架構師是怎樣開展工作的,他/她需要對接上下游哪些角色,以什麼作為工作輸入,最終要對外輸出什麼產物。這些內容既有助於我們跟架構崗同事更好的協作,也可以作為是否往架構轉型的參考,接下來我們一起揭開架構師的... ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...