詳解Go語(yǔ)言如何實(shí)現(xiàn)字符串切片反轉(zhuǎn)函數(shù)
Python 中的 reverse 函數(shù)
Go 語(yǔ)言不像其他語(yǔ)言如 Python,有著內(nèi)置的 reverse() 函數(shù),先來(lái)看一下 Python 中對(duì)于列表的反轉(zhuǎn)方法,然后我們?cè)賮?lái)學(xué)習(xí)如果在 Go 語(yǔ)言中實(shí)現(xiàn)相同的功能。
>>> myList = [2022, 2021, 2008, 2012]
>>> myList.reverse()
>>> print("Reversed List:", myList)
Reversed List: [2012, 2008, 2021, 2022]
>>> 實(shí)現(xiàn)一個(gè) reverse 反轉(zhuǎn)函數(shù)
reverse 算法取一個(gè)數(shù)據(jù)集,并將該數(shù)據(jù)集的值進(jìn)行反轉(zhuǎn),Go 標(biāo)準(zhǔn)的 sort 包并沒(méi)有內(nèi)置的方法來(lái)反轉(zhuǎn)一個(gè)切片。
利用兩個(gè)切片實(shí)現(xiàn)
設(shè)計(jì)思想:
- 確定切片長(zhǎng)度
- 獲取最后一個(gè)元素
- 以相反的順序在新切片中添加最后一個(gè)元素到第一個(gè)位置
package main
import "fmt"
func main() {
s := []string{"hello", "foo", "bar", "go", "abc", "zzz"}
// 定義新的反轉(zhuǎn)切片
reverseOfS := []string{}
// 遍歷原切片 s
for i := range s {
reverseOfS = append(reverseOfS, s[len(s)-1-i])
}
fmt.Println(reverseOfS)
}運(yùn)行結(jié)果:
[zzz abc go bar foo hello]
顯然,這種方式會(huì)額外花費(fèi)一個(gè)相同空間的切片,空間復(fù)雜度為 O(n)。
前后兩兩原地交換
我們可以寫(xiě)一個(gè)簡(jiǎn)易的 reverse 函數(shù)來(lái)進(jìn)行數(shù)據(jù)的反轉(zhuǎn),通過(guò)循環(huán)原切片的一半,然后依次與對(duì)應(yīng)的元素進(jìn)行交換,比如::
func reverse(s []string) []string {
for i := 0; i < len(s)/2; i++ {
j := len(s) - i - 1
s[i], s[j] = s[j], s[i]
}
return s
}這個(gè)函數(shù)可以通過(guò)更簡(jiǎn)短的實(shí)現(xiàn),通過(guò) Go 內(nèi)部的操作進(jìn)行循環(huán):
package main
import "fmt"
func reverse(s []string) []string {
for i, j := 0, len(s)-1; i < j; i, j = i+1, j-1 {
s[i], s[j] = s[j], s[i]
}
return s
}
func main() {
s := []string{"hello", "foo", "bar", "go", "abc", "zzz"}
reverseOfS := reverse(s)
fmt.Println(reverseOfS)
}
執(zhí)行結(jié)果:
[zzz abc go bar foo hello]
但是,上面的 reverse 函數(shù)都是通過(guò)切片按值傳遞,其實(shí)我們?cè)谛薷膫鬟f中的 []string 切片,實(shí)際上,可以通過(guò)以下方式進(jìn)一步簡(jiǎn)寫(xiě):
package main
import "fmt"
func reverse(s []string) {
for i := 0; i < len(s)/2; i++ {
j := len(s) - i - 1
s[i], s[j] = s[j], s[i]
}
}
func main() {
s := []string{"hello", "foo", "bar", "go", "abc", "zzz"}
reverse(s)
fmt.Printf("%v\n", s)
}此時(shí),reverse() 函數(shù)不會(huì)返回切片的另一個(gè)引用,此時(shí)的交換就是就地進(jìn)行,此時(shí)更像文章開(kāi)頭 Python 中的 reverse() 函數(shù)。
反轉(zhuǎn)為原切片的副本
如果我們要返回切片的反轉(zhuǎn)的副本,reverse 函數(shù)就可以這樣寫(xiě):
package main
import "fmt"
func reverse(s []string) []string {
newS := make([]string, len(s))
for i, j := 0, len(s)-1; i <= j; i, j = i+1, j-1 {
newS[i], newS[j] = s[j], s[i]
}
return newS
}
func main() {
s := []string{"hello", "foo", "bar", "go", "abc", "zzz"}
fmt.Printf("原字符串切片:%v\n", s)
fmt.Printf("反轉(zhuǎn)后的切片:%v\n", reverse(s))
}運(yùn)行結(jié)果:
原字符串切片:[hello foo bar go abc zzz]
反轉(zhuǎn)后的切片:[zzz abc go bar foo hello]
可以看到,原切片是沒(méi)有變化的。
當(dāng)然,因?yàn)槲覀儧](méi)有就地修改原切片,因此又可以回到最初的方法 append,看代碼:
func reverse(s []string) []string {
newS := make([]string, 0, len(s))
for i := len(s)-1; i >= 0; i-- {
newS = append(newS, s[i])
}
return newS
}運(yùn)行結(jié)果圖如下:

總結(jié)
本文通過(guò) Python 中的 reverse() 函數(shù)的一個(gè)示例,引發(fā)出一個(gè)思考:Go 語(yǔ)言中有沒(méi)有類(lèi)似的反轉(zhuǎn)函數(shù)?
然后通過(guò)幾種方式實(shí)現(xiàn)同樣的字符串切片的反轉(zhuǎn)功能,并通過(guò)借助額外空間和就地反轉(zhuǎn)兩種方式實(shí)現(xiàn)了功能相同 reverse 函數(shù),其實(shí)類(lèi)似的反轉(zhuǎn)思想也可以用于字符串或者鏈表反轉(zhuǎn)等其他數(shù)據(jù)結(jié)構(gòu)。
到此這篇關(guān)于詳解Go語(yǔ)言如何實(shí)現(xiàn)字符串切片反轉(zhuǎn)函數(shù)的文章就介紹到這了,更多相關(guān)Go語(yǔ)言字符串切片反轉(zhuǎn)函數(shù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Beego中ORM操作各類(lèi)數(shù)據(jù)庫(kù)連接方式詳細(xì)示例
這篇文章主要為大家介紹了Beego中ORM操作各類(lèi)數(shù)據(jù)庫(kù)連接方式詳細(xì)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步早日升職加薪2022-04-04
Windows下在CMD下執(zhí)行Go出現(xiàn)中文亂碼的解決方法
在cmd下運(yùn)行g(shù)o程序或者是GOLAND的Terminal下運(yùn)行g(shù)o程序會(huì)出現(xiàn)中文亂碼的情況。本文就詳細(xì)的介紹下解決方法,具有一定的參考價(jià)值,感興趣的可以了解一下2021-12-12
Go語(yǔ)言實(shí)現(xiàn)機(jī)器大小端判斷代碼分享
這篇文章主要介紹了Go語(yǔ)言實(shí)現(xiàn)機(jī)器大小端判斷代碼分享,本文直接給出實(shí)現(xiàn)代碼,需要的朋友可以參考下2014-10-10
Go到底能不能實(shí)現(xiàn)安全的雙檢鎖(推薦)
這篇文章主要介紹了Go到底能不能實(shí)現(xiàn)安全的雙檢鎖,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-05-05
一文詳細(xì)介紹golang中.()類(lèi)型斷言的使用方法
Golang是一門(mén)非常流行的編程語(yǔ)言,在很多領(lǐng)域都有著廣泛的應(yīng)用,在開(kāi)發(fā)過(guò)程中,很多時(shí)候我們需要將函數(shù)作為參數(shù)傳遞給其他函數(shù),這時(shí)候就需要用到golang中的.()用法,本文將詳細(xì)介紹golang中.()的使用方法,需要的朋友可以參考下2023-08-08

