Java 字符串是不可變的到底是什么意思?
在進(jìn)一步探討不變性之前,讓我們String在得出任何結(jié)論之前先了解一下類及其功能。
這是如何String工作的:
String str = 'kNowledge';
與往常一樣,這將創(chuàng)建一個(gè)包含的字符串”kNowledge”并為其指定一個(gè)reference str。很簡(jiǎn)單?讓我們執(zhí)行更多功能:
String s = str; // assigns a new reference to the same string 'kNowledge'
讓我們看看下面的語句是如何工作的:
str = str.concat(' base');
這會(huì)將一個(gè)字符串附加' base'到str。但是,等等,由于String對(duì)象是不可變的,這怎么可能呢?令你驚訝的是。
當(dāng)執(zhí)行以上語句時(shí),VM接受的值String str,即'kNowledge'和' base',為我們提供值'kNowledge base'。現(xiàn)在,由于Strings是不可變的,因此VM無法將此值分配給str,因此它創(chuàng)建了一個(gè)新String對(duì)象,為其提供了一個(gè)值'kNowledge base',并為其提供了一個(gè)引用str。
這里要注意的重要一點(diǎn)是,盡管String對(duì)象是不可變的,但其引用變量卻不是。這就是為什么在上面的示例中,引用是指新形成的String對(duì)象。
至此,在上面的示例中,我們有兩個(gè)String對(duì)象:第一個(gè)對(duì)象是我們用value創(chuàng)建的,'kNowledge'由指向s,第二個(gè)對(duì)象是'kNowledge base'由指向str。但是,從技術(shù)上講,我們有三個(gè)String對(duì)象,第三個(gè)對(duì)象'base'是concat語句中的文字。
有關(guān)字符串和內(nèi)存使用情況的重要事實(shí)如果沒有其他參考s該”kNowledge”怎么辦?我們會(huì)失去那個(gè)String。但是,它仍然存在,但由于沒有參考文獻(xiàn)而被認(rèn)為丟失。再看下面的一個(gè)例子
String s1 = 'java';s1.concat(' rules');System.out.println('s1 refers to '+s1); // Yes, s1 still refers to 'java'
發(fā)生了什么:
第一行非常簡(jiǎn)單:創(chuàng)建一個(gè)新的String 'java'并引用s1它。接下來,VM創(chuàng)建另一個(gè)new String 'java rules',但沒有任何引用。因此,第二個(gè)String瞬間丟失。我們無法達(dá)到。參考變量s1仍然引用原始變量String 'java'。
應(yīng)用于String對(duì)象以對(duì)其進(jìn)行修改的幾乎每種方法都會(huì)創(chuàng)建新String對(duì)象。那么,這些String物體在哪里呢?嗯,這些存在于內(nèi)存中,任何編程語言的主要目標(biāo)之一就是有效利用內(nèi)存。
隨著應(yīng)用程序的增長(zhǎng),文字通常String會(huì)占用很大的內(nèi)存區(qū)域,這甚至可能導(dǎo)致冗余。因此,為了提高Java的效率,JVM預(yù)留了一個(gè)特殊的內(nèi)存區(qū)域,稱為“字符串常量池”。
當(dāng)編譯器看到String文字時(shí),它將String在池中查找。如果找到匹配項(xiàng),則對(duì)新文字的引用將指向現(xiàn)有文件,String并且不會(huì)String創(chuàng)建新對(duì)象。現(xiàn)有的String只是另外一個(gè)參考。這是使String對(duì)象不可變的要點(diǎn):
在String常量池中,一個(gè)String對(duì)象可能具有一個(gè)或多個(gè)引用。如果幾個(gè)引用指向同一對(duì)象String甚至不知道它,那么如果其中一個(gè)引用修改了該String值,那將是不好的。這就是為什么String對(duì)象是不可變的。
好吧,現(xiàn)在你可以說,如果有人重寫了String類的功能怎么辦?就是這樣的原因的String類被標(biāo)記final,這樣沒有人可以凌駕于其方法的行為。
解決方法我在不可變字符串上編寫了以下代碼。
public class ImmutableStrings { public static void main(String[] args) {testmethod(); } private static void testmethod() {String a = "a";System.out.println("a 1-->" + a);a = "ty";System.out.println("a 2-->" + a); }}
輸出:
a 1-->a a 2-->ty
此處變量的值a已更改(許多人說不能更改不可變對(duì)象的內(nèi)容)。但是,一句話到底是什么String是不變的呢?
相關(guān)文章:
1. 如何用筆記本上的apache做微信開發(fā)的服務(wù)器2. 冒昧問一下,我這php代碼哪里出錯(cuò)了???3. php - 淘寶訂單拆單表設(shè)計(jì)4. mysql優(yōu)化 - MySQL如何為配置表建立索引?5. MySQL主鍵沖突時(shí)的更新操作和替換操作在功能上有什么差別(如圖)6. windows誤人子弟啊7. 實(shí)現(xiàn)bing搜索工具urlAPI提交8. 數(shù)據(jù)庫 - Mysql的存儲(chǔ)過程真的是個(gè)坑!求助下面的存儲(chǔ)過程哪里錯(cuò)啦,實(shí)在是找不到哪里的問題了。9. 我在網(wǎng)址中輸入localhost/abc.php顯示的是not found是為什么呢?10. 關(guān)于mysql聯(lián)合查詢一對(duì)多的顯示結(jié)果問題
