作者iceman5566 (iceman5566)
看板java
標題[問題] 關於 Java 的 call by value/reference
時間Sun Nov 1 23:42:09 2020
想請問一下
String s1 = "I learn";
s1 += " Java";
String s2 = "I";
s2 += " learn Java";
System.out.println(s1 == s2); //false
這個是 false 我可以理解,兩個的記憶體位置不同,因此不會是 true,
但我想問的是
String s1 = "I";
String s2 = "I";
System.out.println(s1 == s2);
既然如剛剛所說 Java 在定義變數時,記憶體位置不相同,應該要是 false,
那為何這邊是 true,在沒有二度賦值時又變成了 call by value?
還是說第一個例子的原理 不算是 call by reference?
剛開始學 Java,抱歉問題有點新手
--
※ 發信站: 批踢踢實業坊(ptt.cc), 來自: 118.167.24.244 (臺灣)
※ 文章網址: https://webptt.com/m.aspx?n=bbs/java/M.1604245331.A.635.html
1F:推 LPH66: 常數字串會統一存在你的 class 裡, 取用時會直接拉出來用 11/02 05:44
2F:→ LPH66: 這又被叫做 string pool 11/02 05:46
3F:→ LPH66: 一樓不太精確, 應該要說是字串實字 (string literal) 11/02 05:47
4F:→ LPH66: 也就是用 "" 夾起來的字串 11/02 05:47
5F:→ x94fujo6: 只有object是用reference 11/02 05:48
那如果是 call by value 第一個不應該是 false 吧
既然都是 call by value,不管字串做什麼樣的拼湊,如果 value 一樣那應該不會是 false?
※ 編輯: iceman5566 (61.222.53.63 臺灣), 11/02/2020 14:42:49
6F:→ ssccg: 你這兩段都沒有call,哪來的call by value/reference?? 11/02 15:00
7F:→ ssccg: 你應該搞錯名詞了,這是primitive/reference type還有 == 11/02 15:02
8F:→ ssccg: operator定義的問題 11/02 15:03
9F:→ ssccg: 基本上是一樓說的,不過不是存在class裡,是JVM load class 11/02 15:10
10F:→ ssccg: 的時候會判斷如果string literal已經在constant pool(記憶 11/02 15:10
11F:→ ssccg: 體中)中,就會重用pool中的,不會再建新的String 11/02 15:12
12F:→ ssccg: 會這樣做是因為String是immutable,共用object沒有風險又省 11/02 15:14
感謝大大 我懂了!
※ 編輯: iceman5566 (61.222.53.63 臺灣), 11/02/2020 15:18:03
13F:推 LPH66: 嗯, 嚴格來說並不是物件本身存在 class 中, 而是該字串內容 11/03 04:34
14F:→ LPH66: 實際在執行時 JVM 還是會看還有沒有其他字串也有共用的 11/03 04:35
15F:→ protoss: 應該是"I"是個字串常數會放在一個string pool...所以就只 11/11 01:40
16F:→ protoss: 有這麼一個物件...所以你的比較才會是true... 11/11 01:40
17F:→ protoss: 而上面那個部分...你就可以把它認為在string pool有右手 11/11 01:41
18F:→ protoss: 邊四個字串常數...s1和s1在做append時其實是另外產生copy 11/11 01:42
19F:→ protoss: 去做append... 11/11 01:42