詳解為什么說Golang中的字符串類型不能修改
在接觸Go這么語(yǔ)言,可能你經(jīng)常會(huì)聽到這樣一句話。對(duì)于字符串不能修改,可能你很納悶,日常開發(fā)中我們對(duì)字符串進(jìn)行修改也是很正常的,為什么又說Go中的字符串不能進(jìn)行修改呢?
本文就來(lái)通過實(shí)際案例給大家演示,為什么Go中的字符串不能進(jìn)行修改。
在演示這個(gè)問題之前,我們先對(duì)字符串類型的基礎(chǔ)知識(shí)做個(gè)大致的演示,這樣便于大家對(duì)問題的進(jìn)一步了解。
字符串定義
字符串是一種用來(lái)表示字符的數(shù)據(jù)類型。在使用時(shí),使用" "將字符內(nèi)容包含起來(lái)。例如下面的形式:
package?main
import?"fmt"
func?main()?{
????var?str?string?=?"Hello?World!"
}在Go中,字符串通常有三種定義方式:
//?第一種(全量定義) var?變量名稱?string?=?"字符串內(nèi)容" //?類型推導(dǎo) var?變量名稱?=?"字符串內(nèi)容" //?短標(biāo)記(只適用于局部變量) 變量名稱?:=?"字符串內(nèi)容"
字符串的定義,其實(shí)也可以通過字節(jié)的方式。這里羅列的方式是最為常見的方式。
字符串的組成
Go中的字符串符合Unicode[1]標(biāo)準(zhǔn),并且采用UTF-8[2]編碼。字符串底層其實(shí)也是由byte組成(后面會(huì)仔細(xì)講解)。通過下面的示例,打印查看具體的字節(jié)內(nèi)容:
s?:=?"Hello?World!"
for?_,?v?:=?range?s?{
????fmt.Print(v)
????fmt.Print("\t")
}
//?72?101?108?108?111?32?87?111?114?108?100?33上面代碼打印的內(nèi)容,就是每一個(gè)字符所表示的字節(jié)碼。
字符串不能修改
通過上面的大致演示,我們對(duì)字符串有一個(gè)基本的了解。對(duì)于字符串不能修改,可能你很納悶,日常開發(fā)中我們對(duì)字符串進(jìn)行重新賦值也是很正常的,為什么又說Go中的字符串不能進(jìn)行修改呢?
其實(shí)這里要糾正這個(gè)說話,對(duì)于字符串修改并不等價(jià)于重新賦值。開發(fā)中常用的方式,其實(shí)是一種重新賦值的概念。
str?:=?"Hello?World!" //?重新賦值 str?=?"Hello?Go!" //?字符串修改 str[0]?=?"I"
通常聽到的不能修改,其實(shí)就是指的上面代碼的第二種方式。并且通過這種方式修改會(huì)報(bào)錯(cuò)::cannot assign to s[0] (value of type byte)
回歸正題,為什么Go中的字符串不能通過下標(biāo)的方式來(lái)進(jìn)行修改呢? 這是因?yàn)镚o中的字符串的數(shù)據(jù)結(jié)構(gòu)體是由一個(gè)指針和長(zhǎng)度組成的結(jié)構(gòu)體,該指針指向的一個(gè)切片才是真正的字符串值。Go中源碼有這樣一段定義:
type?stringStruct?struct?{
????str?unsafe.Pointer?//?指向一個(gè)byte類型的切片指針
????len?int?//?字符串的長(zhǎng)度
}
正是因?yàn)榈讓邮且粋€(gè)[]byte類型的切片,當(dāng)我們使用下標(biāo)的方式去修改值,這時(shí)候?qū)⒁粋€(gè)字符內(nèi)容賦值給byte類型,肯定是不允許的。但是我們可以通過下標(biāo)的方式去訪問對(duì)應(yīng)的byte值。
fmt.Println(s[0])?//?output:72
那我們要想通過下標(biāo)的方式去修改值該怎么辦呢?這時(shí)候,就需要通過切片的方式來(lái)定義,然后在轉(zhuǎn)成字符串。
package?main
import?(??
????"fmt"
)
func?main()?{??
?????s1?:=?[]byte{72,?101,?108,?108,?111,?32,?87,?111,?114,?108,?100,?33}
????fmt.Println(string(s1))
????//?將"H"修改為l
????s1[0]?=?108
????fmt.Println(string(s1))
}
//?output:
Hello?World!
lello?World!字符串的賦值
上面分析了為什么字符串不能使用下標(biāo)去賦值,回過來(lái)解答一下日常開發(fā)中的賦值方式。
package?main
import?(??
????"fmt"
)
func?main()?{
????//?聲明一個(gè)字符串,并給與初始值
????s?:=?"Hello?World!"
????//?對(duì)變量?s?進(jìn)行重新賦值
????s?:=?"Hello?Go!"
}那為什么這種場(chǎng)景下又可以給字符串重新賦值呢? 這是因?yàn)椋贕o的底層其實(shí)是新創(chuàng)建了一個(gè)[]byte{}類型的切片,將變量s中的指針指向了新的內(nèi)存空間地址(也就是這里的Hello Go!)。原有的Hello World!內(nèi)存空間會(huì)隨著垃圾回收機(jī)制被回收掉。

為什么這么設(shè)計(jì)
可能大家都會(huì)考慮到,為什么一個(gè)普通的字符串要設(shè)計(jì)這么復(fù)雜,還需要使用指針。暫時(shí)沒找到官方文檔的說明,
1. 個(gè)人猜想,當(dāng)遇到一個(gè)非常長(zhǎng)的字符時(shí),這樣做使得string變得非常輕量,可以很方便的進(jìn)行傳遞而不用擔(dān)心內(nèi)存拷貝。雖然在Go中,不管是引用類型還是值類型參數(shù)傳遞都是值傳遞。但指針明顯比值傳遞更節(jié)省內(nèi)存。
到此這篇關(guān)于詳解為什么說Golang中的字符串類型不能修改的文章就介紹到這了,更多相關(guān)Golang字符串類型內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
關(guān)于golang中map使用的幾點(diǎn)注意事項(xiàng)總結(jié)(強(qiáng)烈推薦!)
map是一種無(wú)序的基于key-value的數(shù)據(jù)結(jié)構(gòu),Go語(yǔ)言中的map是引用類型,必須初始化才能使用,下面這篇文章主要給大家介紹了關(guān)于golang中map使用的幾點(diǎn)注意事項(xiàng),需要的朋友可以參考下2023-01-01
Centos下搭建golang環(huán)境及vim高亮Go關(guān)鍵字設(shè)置的方法
這篇文章先給大家詳細(xì)介紹了在Centos下搭建golang環(huán)境的步驟,大家按照下面的方法就可以自己搭建golang環(huán)境,搭建完成后又給大家介紹了vim高亮Go關(guān)鍵字設(shè)置的方法,文中通過示例代碼介紹的很詳細(xì),有需要的朋友們可以參考借鑒,下面來(lái)一起看看吧。2016-11-11
golang架構(gòu)設(shè)計(jì)開閉原則手寫實(shí)現(xiàn)
這篇文章主要為大家介紹了golang架構(gòu)設(shè)計(jì)開閉原則手寫實(shí)例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07
7分鐘讀懂Go的臨時(shí)對(duì)象池pool以及其應(yīng)用場(chǎng)景
這篇文章主要給大家介紹了關(guān)于如何通過7分鐘讀懂Go的臨時(shí)對(duì)象池pool以及其應(yīng)用場(chǎng)景的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或使用Go具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起看看吧2018-11-11
go module構(gòu)建項(xiàng)目的實(shí)現(xiàn)
本文主要介紹了go module構(gòu)建項(xiàng)目的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-03-03
加速開發(fā):使用Go語(yǔ)言和Gin框架構(gòu)建Web項(xiàng)目的利器
Go語(yǔ)言和Gin框架是構(gòu)建高性能Web項(xiàng)目的利器,Go語(yǔ)言的簡(jiǎn)潔性和并發(fā)性,以及Gin框架的輕量級(jí)和快速路由能力,使開發(fā)者能夠快速構(gòu)建可靠的Web應(yīng)用程序,需要的朋友可以參考下2023-09-09

