溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Golang字符串類型為什么不能修改

發(fā)布時(shí)間:2022-11-22 09:32:18 來源:億速云 閱讀:111 作者:iii 欄目:編程語言

這篇“Golang字符串類型為什么不能修改”文章的知識(shí)點(diǎn)大部分人都不太理解,所以小編給大家總結(jié)了以下內(nèi)容,內(nèi)容詳細(xì),步驟清晰,具有一定的借鑒價(jià)值,希望大家閱讀完這篇文章能有所收獲,下面我們一起來看看這篇“Golang字符串類型為什么不能修改”文章吧。

字符串定義

字符串是一種用來表示字符的數(shù)據(jù)類型。在使用時(shí),使用” “將字符內(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 組成 (后面會(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)的方式來進(jìn)行修改呢?

這是因?yàn)?Go中的字符串的數(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)度
}

Golang字符串類型為什么不能修改

正是因?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í)候,就需要通過切片的方式來定義,然后在轉(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() {
    // 聲明一個(gè)字符串,并給與初始值
    s := "Hello World!"
    // 對(duì)變量 s 進(jìn)行重新賦值
    s := "Hello Go!"
}

那為什么這種場(chǎng)景下又可以給字符串重新賦值呢?

這是因?yàn)?,?Go 的底層其實(shí)是新創(chuàng)建了一個(gè) [] byte {} 類型的切片,將變量 s 中的指針指向了新的內(nèi)存空間地址 (也就是這里的 Hello Go!)。原有的 Hello World! 內(nèi)存空間會(huì)隨著垃圾回收機(jī)制被回收掉。

Golang字符串類型為什么不能修改

為什么這么設(shè)計(jì)

可能大家都會(huì)考慮到,為什么一個(gè)普通的字符串要設(shè)計(jì)這么復(fù)雜,還需要使用指針。暫時(shí)沒找到官方文檔的說明,

個(gè)人猜想,當(dāng)遇到一個(gè)非常長(zhǎng)的字符時(shí),這樣做使得 string 變得非常輕量,可以很方便的進(jìn)行傳遞而不用擔(dān)心內(nèi)存拷貝。雖然在 Go 中,不管是引用類型還是值類型參數(shù)傳遞都是值傳遞。但指針明顯比值傳遞更節(jié)省內(nèi)存。

以上就是關(guān)于“Golang字符串類型為什么不能修改”這篇文章的內(nèi)容,相信大家都有了一定的了解,希望小編分享的內(nèi)容對(duì)大家有幫助,若想了解更多相關(guān)的知識(shí)內(nèi)容,請(qǐng)關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI