通過案例簡單聊聊為什么說Go中的字符串是不能被修改的
本文將通過實(shí)際案例給大家演示,為什么Go中的字符串不能進(jìn)行修改。
在演示這個問題之前,我們先對字符串類型的基礎(chǔ)知識做個大致的演示,這樣便于大家對問題的進(jìn)一步了解。
字符串定義
字符串是一種用來表示字符的數(shù)據(jù)類型。在使用時,使用" "將字符內(nè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標(biāo)準(zhǔn),并且采用UTF-8編碼。字符串底層其實(shí)也是由byte組成(后面會仔細(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)容,就是每一個字符所表示的字節(jié)碼。
字符串不能修改
通過上面的大致演示,我們對字符串有一個基本的了解。對于字符串不能修改,可能你很納悶,日常開發(fā)中我們對字符串進(jìn)行重新賦值也是很正常的,為什么又說Go中的字符串不能進(jìn)行修改呢?
其實(shí)這里要糾正這個說話,對于字符串修改并不等價于重新賦值。開發(fā)中常用的方式,其實(shí)是一種重新賦值的概念。
str := "Hello World!" // 重新賦值 str = "Hello Go!" // 字符串修改 str[0] = "I"
通常聽到的不能修改,其實(shí)就是指的上面代碼的第二種方式。并且通過這種方式修改會報錯::cannot assign to s[0] (value of type byte)
回歸正題,為什么Go中的字符串不能通過下標(biāo)的方式來進(jìn)行修改呢? 這是因?yàn)?code>Go中的字符串的數(shù)據(jù)結(jié)構(gòu)體是由一個指針和長度組成的結(jié)構(gòu)體,該指針指向的一個切片才是真正的字符串值。Go中源碼有這樣一段定義:
type stringStruct struct {
str unsafe.Pointer // 指向一個byte類型的切片指針
len int // 字符串的長度
}
正是因?yàn)榈讓邮且粋€[]byte類型的切片,當(dāng)我們使用下標(biāo)的方式去修改值,這時候?qū)⒁粋€字符內(nèi)容賦值給byte類型,肯定是不允許的。但是我們可以通過下標(biāo)的方式去訪問對應(yīng)的byte值。
fmt.Println(s[0]) // output:72
那我們要想通過下標(biāo)的方式去修改值該怎么辦呢?這時候,就需要通過切片的方式來定義,然后在轉(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)去賦值,回過來解答一下日常開發(fā)中的賦值方式。
package main
import (
"fmt"
)
func main() {
// 聲明一個字符串,并給與初始值
s := "Hello World!"
// 對變量 s 進(jìn)行重新賦值
s := "Hello Go!"
}那為什么這種場景下又可以給字符串重新賦值呢? 這是因?yàn)?,在Go的底層其實(shí)是新創(chuàng)建了一個[]byte{}類型的切片,將變量s中的指針指向了新的內(nèi)存空間地址(也就是這里的Hello Go!)。原有的Hello World!內(nèi)存空間會隨著垃圾回收機(jī)制被回收掉。

為什么這么設(shè)計
可能大家都會考慮到,為什么一個普通的字符串要設(shè)計這么復(fù)雜,還需要使用指針。暫時沒找到官方文檔的說明,
- 個人猜想,當(dāng)遇到一個非常長的字符時,這樣做使得string變得非常輕量,可以很方便的進(jìn)行傳遞而不用擔(dān)心內(nèi)存拷貝。雖然在Go中,不管是引用類型還是值類型參數(shù)傳遞都是值傳遞。但指針明顯比值傳遞更節(jié)省內(nèi)存。
以上就是通過案例簡單聊聊為什么說Go中的字符串是不能被修改的的詳細(xì)內(nèi)容,更多關(guān)于Go字符串不能修改的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Go語言O(shè)RM包中使用worm構(gòu)造查詢條件的實(shí)例詳解
worm是一款方便易用的Go語言O(shè)RM庫。worm支Model方式(持結(jié)構(gòu)體字段映射)、原生SQL以及SQLBuilder三種模式來操作數(shù)據(jù)庫,并且Model方式、原生SQL以及SQLBuilder可混合使用,本文通過一些例子來說明如何使用worm來構(gòu)造查詢條件,感興趣的朋友一起看看吧2022-07-07
Golang信號量設(shè)計實(shí)現(xiàn)示例詳解
這篇文章主要為大家介紹了Golang信號量設(shè)計實(shí)現(xiàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-08-08
Go語言網(wǎng)站使用異步編程和Goroutine提高Web的性能
作為一門現(xiàn)代化編程語言,Go語言提供了強(qiáng)大的異步編程能力,使得程序員可以以更高效的方式處理并發(fā)任務(wù),在Go語言中,使用Goroutine在單個進(jìn)程中實(shí)現(xiàn)多任務(wù)并行處理,以及如何使用協(xié)程池來進(jìn)一步提高Web服務(wù)器的處理能力,2024-01-01
Go語言實(shí)現(xiàn)讀取文件的方式總結(jié)
這篇文章主要為大家詳細(xì)介紹了Go語言實(shí)現(xiàn)讀取文件的幾個方式,文中的示例代碼講解詳細(xì),對我們學(xué)習(xí)Go語言有一定的幫助,感興趣的小伙伴可以收藏一下2023-04-04
golang常用庫之gorilla/mux-http路由庫使用詳解
這篇文章主要介紹了golang常用庫之gorilla/mux-http路由庫使用,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-10-10
Go語言中的定時器原理與實(shí)戰(zhàn)應(yīng)用
在Go語言中,Timer和Ticker是處理定時任務(wù)的重要工具,Timer用于一次性事件,而Ticker則用于周期性事件,本文詳細(xì)介紹了這兩種定時器的創(chuàng)建、使用和停止方法,并通過實(shí)際案例展示了它們在監(jiān)控日志、檢查系統(tǒng)狀態(tài)等方面的應(yīng)用2024-10-10
Golang 端口復(fù)用測試的實(shí)現(xiàn)
這篇文章主要介紹了Golang 端口復(fù)用測試的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03

