觀察情況一(字符串實例直接賦值給string類型的變量):
分析:
通過調試看出,兩個變量存儲的內存地址是一樣的,這個內存地址其實指向的是字符串常量區
圖解:
原理:
創建一個字符串對象,系統會先掃描常量區有沒有相同值的字符串,如果有,就直接返回常量區對應的地址 。
觀察情況二(通過 new 關鍵字實例化string類型的對象):
分析:
此時為什麼使用new關鍵字創建字符串實例,怎麼內存地址沒有像情況一中內存地址都一樣。原因是new關鍵字的本質促使。
new的作用:
1.開辟堆內存空間或者常量區
2.創建對象
3.調用構造函數
4.返回開辟的內存地址
由此分析new,是開辟了新的內存區域,自然返回的內存地址也不會一樣。
觀察情況三:
思考:
從正常邏輯分析,a和b指向了同一個對象的引用,案例b="bb";後變量a的值也應該發生改變才對。由於是根據字符串特性的不同,這裡才使a得值沒有發生變化。
圖解:
原理:
如果對一個字符串類型的變量的內容進行修改,其實是無法修改變量存儲的字符串本身的,而是根據這個新的字符串,去掃描常量區,如果沒有就開辟新的空間存儲這個字符串,並返回新的內存空間地址。
字符串本質總結:
1. 創建一個字符串,系統會先掃描常量區有沒有相同值的字符串,如果有,就直接返回常量區對應的地址 。
2. 如果對一個字符串類型的變量的內容進行修改,其實是無法修改變量存儲的字符串本身的,而是根據這個新的字符串,去掃描常量區,如果沒有就開辟新的空間存儲這個字符串,並返回新的內存空間地址。
3.字符串具有恆定性(值不能改)的特點,一旦在常量區中創建,生命周期是隨著應用程序結束而釋放。
4.字符串存儲在常量區中,作用域是一個公共的區域。
面試實戰題目:
不是說字符串是不可變的嗎?string s="abc";s="123"不就是變了嗎?
答:
在這裡,變量s的存儲的值確實是由"abc"變成"123"。但是這僅僅改變的是變量s所指向的不同引用。"abc"是一個字符串實例對象,
"abc"的創建首先會掃描堆區域中的常量區,有沒有"abc",如果有則直接返回地址,如果沒有則開辟一塊新的區域,並返回新的地址。
根據字符串的恆定性,一旦在常量區中創建,生命周期是隨著應用程序結束而釋放。
所以當創建"123"時,沒有改變"abc"。