Java 字符串是不可變的。到底是什么意思?
在進(jìn)一步探討不變性之前,讓我們String在得出任何結(jié)論之前先了解一下類及其功能。
這是如何String工作的:
String str = 'kNowledge';
與往常一樣,這將創(chuàng)建一個(gè)包含的字符串”kNowledge”并為其指定一個(gè)reference str。很簡單?讓我們執(zhí)行更多功能:
String s = str; // assigns a new reference to the same string 'kNowledge'
讓我們看看下面的語句是如何工作的:
str = str.concat(' base');
這會(huì)將一個(gè)字符串附加' base'到str。但是,等等,由于String對象是不可變的,這怎么可能呢?令你驚訝的是。
當(dāng)執(zhí)行以上語句時(shí),VM接受的值String str,即'kNowledge'和' base',為我們提供值'kNowledge base'。現(xiàn)在,由于Strings是不可變的,因此VM無法將此值分配給str,因此它創(chuàng)建了一個(gè)新String對象,為其提供了一個(gè)值'kNowledge base',并為其提供了一個(gè)引用str。
這里要注意的重要一點(diǎn)是,盡管String對象是不可變的,但其引用變量卻不是。這就是為什么在上面的示例中,引用是指新形成的String對象。
至此,在上面的示例中,我們有兩個(gè)String對象:第一個(gè)對象是我們用value創(chuàng)建的,'kNowledge'由指向s,第二個(gè)對象是'kNowledge base'由指向str。但是,從技術(shù)上講,我們有三個(gè)String對象,第三個(gè)對象'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ā)生了什么:
第一行非常簡單:創(chuàng)建一個(gè)新的String 'java'并引用s1它。接下來,VM創(chuàng)建另一個(gè)new String 'java rules',但沒有任何引用。因此,第二個(gè)String瞬間丟失。我們無法達(dá)到。參考變量s1仍然引用原始變量String 'java'。
應(yīng)用于String對象以對其進(jìn)行修改的幾乎每種方法都會(huì)創(chuàng)建新String對象。那么,這些String物體在哪里呢?嗯,這些存在于內(nèi)存中,任何編程語言的主要目標(biāo)之一就是有效利用內(nèi)存。
隨著應(yīng)用程序的增長,文字通常String會(huì)占用很大的內(nèi)存區(qū)域,這甚至可能導(dǎo)致冗余。因此,為了提高Java的效率,JVM預(yù)留了一個(gè)特殊的內(nèi)存區(qū)域,稱為“字符串常量池”。
當(dāng)編譯器看到String文字時(shí),它將String在池中查找。如果找到匹配項(xiàng),則對新文字的引用將指向現(xiàn)有文件,String并且不會(huì)String創(chuàng)建新對象。現(xiàn)有的String只是另外一個(gè)參考。這是使String對象不可變的要點(diǎn):
在String常量池中,一個(gè)String對象可能具有一個(gè)或多個(gè)引用。如果幾個(gè)引用指向同一對象String甚至不知道它,那么如果其中一個(gè)引用修改了該String值,那將是不好的。這就是為什么String對象是不可變的。
好吧,現(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已更改(許多人說不能更改不可變對象的內(nèi)容)。但是,一句話到底是什么String是不變的呢?
相關(guān)文章:
1. 如何解決docker宿主機(jī)無法訪問容器中的服務(wù)?2. javascript - 如何使用nodejs 將.html 文件轉(zhuǎn)化成canvas3. angular.js - 輸入郵箱地址之后, 如何使其自動(dòng)在末尾添加分號?4. javascript - html5的data屬性怎么指定一個(gè)function函數(shù)呢?5. python - Scrapy存在內(nèi)存泄漏的問題。6. 在mac下出現(xiàn)了兩個(gè)docker環(huán)境7. docker-compose中volumes的問題8. java如何生成token?9. javascript - 后臺(tái)管理系統(tǒng)左側(cè)折疊導(dǎo)航欄數(shù)據(jù)較多,怎么樣直接通過搜索去定位到具體某一個(gè)菜單項(xiàng)位置,并展開當(dāng)前菜單10. javascript - Web微信聊天輸入框解決方案
