go1.17版本在8月16號發(fā)布了,新增的功能和變更如下:
1. 編譯優(yōu)化
go1.17將使用棧傳遞參數(shù)和返回值替換為使用寄存器低零。實現(xiàn)性能提升5%,最終生成的二進制包大小減少2%绍在。
該優(yōu)化目前支持Linux凄诞、macOS强饮、Windows的64位X86架構(gòu)由桌。官方表示后續(xù)會支持更多架構(gòu)。
2. 跨平臺支持
支持windows系統(tǒng)64位ARM架構(gòu)
3. go module改變
新增了 pruned module graphs 功能邮丰,當go.mod文件中指定了go 1.17或者更高版本行您,且依賴的包同樣是go 1.17或者更高版本,go.mod中只保留直接依賴剪廉。
4. 新增語言特性
1. 新增unsafe.Add方法娃循,方便指針運算,是Pointer(uintptr(ptr) + uintptr(len)
的簡化
func Add(ptr Pointer, len IntegerType) Pointer
2. 新增unsafe.Slice方法斗蒋,方便將指針轉(zhuǎn)換為slice捌斧,是(*[len]ArbitraryType)(unsafe.Pointer(ptr))[:]
的簡化
func Slice(ptr *ArbitraryType, len IntegerType) []ArbitraryType
下面代碼為它的使用示例:
package main
import (
"fmt"
"unsafe"
"reflect"
)
func main() {
s := []int{1,2,3}
fmt.Println(s)
printSliceHeader(&s)
s1 := unsafe.Slice(&s[0], 3)
fmt.Println(s1)
printSliceHeader(&s1)
}
func printSliceHeader(s *[]int) {
header := (*reflect.SliceHeader)(unsafe.Pointer(s))
fmt.Println(header)
}
運行結(jié)果為:
[1 2 3]
&{824634818560 3 3}
[1 2 3]
&{824634818560 3 3}
3. 支持從slice到array轉(zhuǎn)換
Converting a slice to an array pointer yields a pointer to the underlying array of the slice. If the length of the slice is less than the length of the array, a run-time panic occurs.
slice轉(zhuǎn)為array指針將生成一個指向slice底層數(shù)組的指針笛质。如果slice的長度小于array的長度,會panic捞蚂。
s := make([]byte, 2, 4)
s0 := (*[0]byte)(s) // s0 != nil
s1 := (*[1]byte)(s[1:]) // &s1[0] == &s[1]
s2 := (*[2]byte)(s) // &s2[0] == &s[0]
s4 := (*[4]byte)(s) // panics: len([4]byte) > len(s)
var t []string
t0 := (*[0]string)(t) // t0 == nil
t1 := (*[1]string)(t) // panics: len([1]string) > len(t)
u := make([]byte, 0)
u0 = (*[0]byte)(u) // u0 != nil
通過例子了解slice轉(zhuǎn)為array的原理:
package main
import (
"fmt"
"reflect"
"unsafe"
)
func main() {
s := []int{1, 2, 3}
fmt.Println("slice中內(nèi)容:", s)
printSliceHeader(&s)
s0 := (*[3]int)(s)
fmt.Println("array中內(nèi)容:", s0)
fmt.Printf("array地址:%p \n", s0)
fmt.Println("")
s0[0] = 0
fmt.Println("1. 改變slice中的元素妇押,array中元素同樣改變")
fmt.Println("slice中內(nèi)容:", s)
fmt.Println("array中內(nèi)容:", s0)
fmt.Println("")
fmt.Println("2. 對slice進行擴容")
s = append(s, 4)
fmt.Println("slice中內(nèi)容:", s)
printSliceHeader(&s)
fmt.Println("array中內(nèi)容:", s0)
fmt.Printf("array地址:%p \n", s0)
fmt.Println("")
fmt.Println("3. array傳參會對數(shù)組中元素進行復制")
arrayParam(*s0)
fmt.Println("array中內(nèi)容:", s0)
}
func printSliceHeader(s *[]int) {
header := (*reflect.SliceHeader)(unsafe.Pointer(s))
//fmt.Println(header)
fmt.Printf("slice底層數(shù)組地址:0x%x \n", header.Data)
}
func arrayParam(s0 [3]int) {
s0[0] = 10
}
運行結(jié)果:
slice中內(nèi)容: [1 2 3]
slice底層數(shù)組地址:0xc000016018
array中內(nèi)容: &[1 2 3]
array地址:0xc000016018
1. 改變slice中的元素,array中元素同樣改變
slice中內(nèi)容: [0 2 3]
array中內(nèi)容: &[0 2 3]
2. 對slice進行擴容
slice中內(nèi)容: [0 2 3 4]
slice底層數(shù)組地址:0xc000078060
array中內(nèi)容: &[0 2 3]
array地址:0xc000016018
3. array傳參會對數(shù)組中元素進行復制
array中內(nèi)容: &[0 2 3]
5. 性能提升和bug修復
1. 提升crypto/x509性能
2. 修復URL Query解析bug姓迅,在1.17版本前敲霍,分號(;)和&一樣可以作為參數(shù)的分隔符號。1.17版本去掉了分號作為分隔符丁存。
6. 泛型前瞻
go預計會在1.18版本正式發(fā)布泛型肩杈,在這之前可以通過一些其他方法來體驗,例如通過編譯go master分支源碼解寝,或者使用官方提供的Play ground扩然。下面這個去重的例子就是在play ground上運行的:
package main
import (
"fmt"
)
type Set[T comparable] map[T]struct{} //comparable限定了類型T必須是可比較類型,這是由于map的key必須是可比較的
func RemoveRep[T comparable](s []T) (s0 []T) {
m := make(Set[T])
var index int
for i := 0; i < len(s); i++ {
if _, ok := m[s[i]]; !ok {
m[s[i]] = struct{}{}
s[index] = s[i]
index++
}
}
return s[:index]
}
type bar struct {
A int
B string
}
func main() {
s := RemoveRep([]int{1, 2, 3, 4, 4, 5, 6, 7, 2, 1})
fmt.Println(s)
s1 := RemoveRep([]string{"a", "b", "c", "c", "d", "e", "a", "b"})
fmt.Println(s1)
s3 := RemoveRep([]bar{{1, "a"}, {2, "b"}, {3, "c"}, {3, "c"}, {4, "d"}, {5, "e"}, {1, "a"}, {2, "b"}})
fmt.Println(s3)
}