成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Java 中的 String 為什么是不可變的?

開發(fā) 后端
眾所周知, 在Java中, String類是不可變的。那么到底什么是不可變的對象呢? 可以這樣認(rèn)為:如果一個對象,在它創(chuàng)建完成之后,不能再改變它的狀態(tài),那么這個對象就是不可變的。不能改變狀態(tài)的意思是,不能改變對象內(nèi)的成員變量,包括 基本數(shù)據(jù)類型的值不能改變,引用類型的變量不能指向其他的對象,引用類型指向的對象的狀態(tài)也不能改變。

什么是不可變對象?

眾所周知, 在Java中, String類是不可變的。那么到底什么是不可變的對象呢? 可以這樣認(rèn)為:如果一個對象,在它創(chuàng)建完成之后,不能再改變它的狀態(tài),那么這個對象就是不可變的。不能改變狀態(tài)的意思是,不能改變對象內(nèi)的成員變量,包括 基本數(shù)據(jù)類型的值不能改變,引用類型的變量不能指向其他的對象,引用類型指向的對象的狀態(tài)也不能改變。

區(qū)分對象和對象的引用

對于Java初學(xué)者, 對于String是不可變對象總是存有疑惑??聪旅娲a:

String s = "ABCabc";
System.out.println("s = " + s);

s = "123456";
System.out.println("s = " + s);

打印結(jié)果為:

s = ABCabc
s = 123456

首先創(chuàng)建一個String對象s,然后讓s的值為“ABCabc”, 然后又讓s的值為“123456”。 從打印結(jié)果可以看出,s的值確實(shí)改變了。那么怎么還說String對象是不可變的呢? 其實(shí)這里存在一個誤區(qū): s只是一個String對象的引用,并不是對象本身。對象在內(nèi)存中是一塊內(nèi)存區(qū),成員變量越多,這塊內(nèi)存區(qū)占的空間越大。引用只是一個4字節(jié)的數(shù)據(jù),里面 存放了它所指向的對象的地址,通過這個地址可以訪問對象。

也就是說,s只是一個引用,它指向了一個具體的對象,當(dāng)s=“123456”; 這句代碼執(zhí)行過之后,又創(chuàng)建了一個新的對象“123456”, 而引用s重新指向了這個心的對象,原來的對象“ABCabc”還在內(nèi)存中存在,并沒有改變。內(nèi)存結(jié)構(gòu)如下圖所示:

Java中的String為什么是不可變的? — String源碼分析

Java和C++的一個不同點(diǎn)是, 在Java中不可能直接操作對象本身,所有的對象都由一個引用指向,必須通過這個引用才能訪問對象本身,包括獲取成員變量的值,改變對象的成員變量,調(diào)用 對象的方法等。而在C++中存在引用,對象和指針三個東西,這三個東西都可以訪問對象。其實(shí),Java中的引用和C++中的指針在概念上是相似的,他們都 是存放的對象在內(nèi)存中的地址值,只是在Java中,引用喪失了部分靈活性,比如Java中的引用不能像C++中的指針那樣進(jìn)行加減運(yùn)算。

為什么String對象是不可變的?

要理解String的不可變性,首先看一下String類中都有哪些成員變量。 在JDK1.6中,String的成員變量有以下幾個:

 

  1. public final class String 
  2.     implements java.io.Serializable, Comparable<String>, CharSequence 
  3.     /** The value is used for character storage. */ 
  4.     private final char value[]; 
  5.  
  6.     /** The offset is the first index of the storage that is used. */ 
  7.     private final int offset; 
  8.  
  9.     /** The count is the number of characters in the String. */ 
  10.     private final int count; 
  11.  
  12.     /** Cache the hash code for the string */ 
  13.     private int hash; // Default to 0 

 

在JDK1.7中,String類做了一些改動,主要是改變了substring方法執(zhí)行時的行為,這和本文的主題不相關(guān)。JDK1.7中String類的主要成員變量就剩下了兩個:

 

  1. public final class String   
  2.     implements java.io.Serializable, Comparable<String>, CharSequence {   
  3.     /** The value is used for character storage. */   
  4.     private final char value[];   
  5.  
  6.     /** Cache the hash code for the string */   
  7.     private int hash; // Default to 0 

由以上的代碼可以看出, 在Java中String類其實(shí)就是對字符數(shù)組的封裝。JDK6中, value是String封裝的數(shù)組,offset是String在這個value數(shù)組中的起始位置,count是String所占的字符的個數(shù)。在 JDK7中,只有一個value變量,也就是value中的所有字符都是屬于String這個對象的。這個改變不影響本文的討論。 除此之外還有一個hash成員變量,是該String對象的哈希值的緩存,這個成員變量也和本文的討論無關(guān)。在Java中,數(shù)組也是對象(可以參考我之前 的文章java中數(shù)組的特性)。 所以value也只是一個引用,它指向一個真正的數(shù)組對象。其實(shí)執(zhí)行了String s = “ABCabc”; 這句代碼之后,真正的內(nèi)存布局應(yīng)該是這樣的:

Java中的String為什么是不可變的? &#8212; String源碼分析

value,offset和count這三個變量都是private的,并且沒有提供setValue, setOffset和setCount等公共方法來修改這些值,所以在String類的外部無法修改String。也就是說一旦初始化就不能修改, 并且在String類的外部不能訪問這三個成員。此外,value,offset和count這三個變量都是final的, 也就是說在String類內(nèi) 部,一旦這三個值初始化了, 也不能被改變。所以可以認(rèn)為String對象是不可變的了。

那么在String中,明明存在一些方法,調(diào)用他們可以得到改變后的值。這些方法包括substring, replace, replaceAll, toLowerCase等。例如如下代碼:

 

  1. String a = "ABCabc";   
  2. System.out.println("a = " + a);   
  3. a = a.replace('A''a');   
  4. System.out.println("a = " + a); 

打印結(jié)果為:

a = ABCabc
a = aBCabc

那么a的值看似改變了,其實(shí)也是同樣的誤區(qū)。再次說明, a只是一個引用, 不是真正的字符串對象,在調(diào)用a.replace(‘A’, ‘a’)時, 方法內(nèi)部創(chuàng)建了一個新的String對象,并把這個心的對象重新賦給了引用a。String中replace方法的源碼可以說明問題:

Java中的String為什么是不可變的? &#8212; String源碼分析

讀者可以自己查看其他方法,都是在方法內(nèi)部重新創(chuàng)建新的String對象,并且返回這個新的對象,原來的對象是不會被改變的。這也是為什么像 replace, substring,toLowerCase等方法都存在返回值的原因。也是為什么像下面這樣調(diào)用不會改變對象的值:

 

  1. String ss = "123456"
  2.  
  3. System.out.println("ss = " + ss); 
  4.  
  5. ss.replace('1''0'); 
  6.  
  7. System.out.println("ss = " + ss); 

 

打印結(jié)果:

ss = 123456
ss = 123456

String對象真的不可變嗎?

從上文可知String的成員變量是private final 的,也就是初始化之后不可改變。那么在這幾個成員中, value比較特殊,因?yàn)樗且粋€引用變量,而不是真正的對象。value是final修飾的,也就是說final不能再指向其他數(shù)組對象,那么我能改變 value指向的數(shù)組嗎? 比如將數(shù)組中的某個位置上的字符變?yōu)橄聞澗€“_”。 至少在我們自己寫的普通代碼中不能夠做到,因?yàn)槲覀兏静荒軌蛟L問到這個value引用,更不能通過這個引用去修改數(shù)組。

那么用什么方式可以訪問私有成員呢? 沒錯,用反射, 可以反射出String對象中的value屬性, 進(jìn)而改變通過獲得的value引用改變數(shù)組的結(jié)構(gòu)。下面是實(shí)例代碼:

 

  1. public static void testReflection() throws Exception { 
  2.  
  3.     //創(chuàng)建字符串"Hello World", 并賦給引用s 
  4.     String s = "Hello World";  
  5.  
  6.     System.out.println("s = " + s); //Hello World 
  7.  
  8.     //獲取String類中的value字段 
  9.     Field valueFieldOfString = String.class.getDeclaredField("value"); 
  10.  
  11.     //改變value屬性的訪問權(quán)限 
  12.     valueFieldOfString.setAccessible(true); 
  13.  
  14.     //獲取s對象上的value屬性的值 
  15.     char[] value = (char[]) valueFieldOfString.get(s); 
  16.  
  17.     //改變value所引用的數(shù)組中的第5個字符 
  18.     value[5] = '_'
  19.  
  20.     System.out.println("s = " + s);  //Hello_World 

 

打印結(jié)果為:

s = Hello World
s = Hello_World

在這個過程中,s始終引用的同一個String對象,但是再反射前后,這個String對象發(fā)生了變化, 也就是說,通過反射是可以修改所謂的“不可變”對象的。但是一般我們不這么做。這個反射的實(shí)例還可以說明一個問題:如果一個對象,他組合的其他對象的狀態(tài) 是可以改變的,那么這個對象很可能不是不可變對象。例如一個Car對象,它組合了一個Wheel對象,雖然這個Wheel對象聲明成了private final 的,但是這個Wheel對象內(nèi)部的狀態(tài)可以改變, 那么就不能很好的保證Car對象不可變。

責(zé)任編輯:王雪燕 來源: 張紀(jì)剛的博客
相關(guān)推薦

2015-03-19 15:04:06

2023-05-29 08:03:41

代碼Go語言

2021-04-07 17:06:55

String Final存儲

2021-01-22 15:31:47

JavaSwitchString

2022-11-26 08:03:57

StringJava

2025-05-19 19:01:14

MCPAPILLM

2024-04-08 07:58:11

Python數(shù)據(jù)類型字符串

2024-09-24 08:36:18

2021-02-01 13:53:53

StringlongJava

2015-02-26 15:06:52

思科

2020-11-02 17:21:07

云計(jì)算

2021-05-19 16:21:16

比特幣加密貨幣貨幣

2023-08-28 08:44:11

NumPyPython

2021-08-26 09:01:35

內(nèi)部Rust可變性

2013-02-25 09:46:22

2022-01-23 22:45:52

物聯(lián)網(wǎng)Java編程語言

2023-06-07 19:17:14

UbuntuKDELinux

2023-10-20 08:18:17

Python數(shù)據(jù)類型

2024-07-01 10:16:55

搜索向量數(shù)據(jù)類型

2014-11-21 10:50:26

JavaString
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 中文在线日韩 | 欧美a区| 午夜精品久久久久久 | 国产在线观看 | 久久久精品黄色 | 亚洲一区二区三区免费在线观看 | 欧美一区二区三区精品免费 | 全免费a级毛片免费看视频免费下 | 日韩美女爱爱 | 日本a在线| 久久久国产一区 | 国产精品一码二码三码在线 | 国产精品一区二区av | 成人av观看 | 国产福利在线看 | 亚洲日本三级 | 一区二区三区中文字幕 | 黄色片免费在线观看 | 国产激情在线播放 | 日韩欧美在线不卡 | 人人射人人插 | 国产精品一二三区在线观看 | 国产精品一区二区不卡 | 国产精品一区久久久 | 日韩在线免费播放 | 一a一片一级一片啪啪 | 亚洲精品成人在线 | 亚洲性视频 | 日本精品一区二区在线观看 | 91精品国产91久久久久久吃药 | 午夜午夜精品一区二区三区文 | 亚洲精品一区二区三区在线观看 | 欧美群妇大交群中文字幕 | 日韩中文av在线 | 羞羞网站在线观看 | 91久久精品一区二区二区 | 欧美激情综合五月色丁香小说 | 日韩欧美手机在线 | 黄色大全免费看 | 在线色网站| 欧洲国产精品视频 |