目錄
- 語言內(nèi)建的資源嵌入支持
- 支持arm64
- go modules的新特性
- GO111MODULE現(xiàn)在默認(rèn)為on
- go build不再更改mod相關(guān)文件
- go install的變化
- 新的GOVCS環(huán)境變量
- 相對(duì)路徑導(dǎo)入不在被允許
- 標(biāo)準(zhǔn)庫(kù)的變化
- testing
- ioutils包已經(jīng)廢棄
- tcp半連接隊(duì)列擴(kuò)容
- 重大更新io/fs
- 其他改進(jìn)
golang1.16也在今天正式發(fā)布了。
原定計(jì)劃是2月1號(hào)年前發(fā)布的,不過遲到也是golang的老傳統(tǒng)了,正好也趁著最后的假期快速預(yù)覽一下golang1.16的新特性吧。
語言內(nèi)建的資源嵌入支持
之前市面上已經(jīng)有很多把今天文件嵌入golang二進(jìn)制程序的工具了,這次golang官方將這一功能加入了embed
標(biāo)準(zhǔn)庫(kù),從語言層面上提供了支持。
我之前以及寫了embed的使用教程,可以看這里。
這兒還有一篇官方推薦的教程。
支持arm64
m1芯片可謂是最近的焦點(diǎn),golang自然也不會(huì)落下。
在golang1.16中官方已經(jīng)支持darwin/arm64
平臺(tái),cgo和編譯成c語言可調(diào)用的動(dòng)態(tài)/靜態(tài)鏈接庫(kù)的功能也已支持。同樣受益的還有bsd家族的arm64版本。
現(xiàn)在可以在新版mac上嘗試golang了。
不過plugin模式的支持仍在進(jìn)行中,想要完整支持arm64還需要一段時(shí)間。
go modules的新特性
本次更新依舊帶來了許多modules的新特性。
GO111MODULE現(xiàn)在默認(rèn)為on
1.16開始默認(rèn)啟用modules,這在1.15的時(shí)候已經(jīng)預(yù)告過了?,F(xiàn)在GO111MODULE的默認(rèn)值為on。
不過golang還是提供了一個(gè)版本的適應(yīng)期,如果你還不習(xí)慣modules,可以把GO111MODULE設(shè)置回auto。在1.17中這個(gè)環(huán)境變量將會(huì)被刪除。
都1202年了,也該學(xué)學(xué)go modules怎么用了。
go build不再更改mod相關(guān)文件
以前的教程里我提到過go build會(huì)自動(dòng)下載依賴,這會(huì)更新mod文件。
現(xiàn)在這一行為被禁止了。想要安裝、更新依賴只能使用go get命令,go build和go test將不會(huì)再做這類工作。
go install的變化
go install在1.16中也有了不小的變化。
首先是通過go install my.module/tool@1.0.0 這樣在module末尾加上版本號(hào),可以在不影響當(dāng)前mod的依賴的情況下安裝golang程序。
go install是未來唯一可以安裝golang程序的命令,go get的編譯安裝功能現(xiàn)在可以靠-d
選項(xiàng)關(guān)閉,而未來編譯安裝功能會(huì)從go get移除。
也就是說go的命令各司其職,不再長(zhǎng)臂管轄了。
新的GOVCS環(huán)境變量
新的GOVCS環(huán)境變量指定了golang用什么版本控制工具下載源代碼。
其格式為:GOVCS=module prefix>:tool name>,[module prefix>:tool name>, ...]
其中module prefix為github.com等,而tool name就是版本控制工具的名字,比如git,svn。
一個(gè)更具體的例子是:GOVCS=github.com:git,evil.com:off,*:git|hg
module prefix也可以用*
通配任何模塊的前綴。
tool name還可以設(shè)置為all和off,all代表允許使用任何可用的工具,而off則表示不允許使用任何版本控制工具。
不過現(xiàn)在設(shè)置為off的模塊的代碼仍然可能會(huì)被下載。
更多的細(xì)節(jié)可以參考go help vcs
。
相對(duì)路徑導(dǎo)入不在被允許
golang1.16開始禁止import導(dǎo)入的模塊以.
開頭,模塊路徑中也不允許出現(xiàn)任何非ASCII字符,所以下面的代碼不再合法:
import (
"./tools/factory"
"../models/user"
"some.pkg.com/殺馬特/音樂工廠"
)
對(duì)非ASCII字符一如既往的不友好,不過也只能按規(guī)矩辦事了。
標(biāo)準(zhǔn)庫(kù)的變化
golang1.16除了對(duì)標(biāo)準(zhǔn)庫(kù)進(jìn)行通常的功能更新和修復(fù),還引入了一些重大變化。
testing
testing包主要的變化是在測(cè)試用例里調(diào)用os.Exit(0)
會(huì)從程序終止變成測(cè)試失敗。
比如這個(gè):
package main
import (
"os"
"testing"
)
func TestXXX(t *testing.T) {
t.Log("exit")
os.Exit(0)
}
現(xiàn)在會(huì)是這樣的輸出:
$ go test -v a_test.go
=== RUN TestXXX
a_test.go:9: exit
--- FAIL: TestXXX (0.00s)
panic: unexpected call to os.Exit(0) during test [recovered]
panic: unexpected call to os.Exit(0) during test
goroutine 18 [running]:
testing.tRunner.func1.2(0x51b920, 0x56cc28)
/usr/local/go/src/testing/testing.go:1144 +0x332
testing.tRunner.func1(0xc000082600)
/usr/local/go/src/testing/testing.go:1147 +0x4b6
panic(0x51b920, 0x56cc28)
/usr/local/go/src/runtime/panic.go:965 +0x1b9
os.Exit(0x0)
/usr/local/go/src/os/proc.go:68 +0x6d
command-line-arguments.TestXXX(0xc000082600)
/tmp/a_test.go:10 +0x76
testing.tRunner(0xc000082600, 0x54df18)
/usr/local/go/src/testing/testing.go:1194 +0xef
created by testing.(*T).Run
/usr/local/go/src/testing/testing.go:1239 +0x2b3
FAIL command-line-arguments 0.004s
FAIL
ioutils包已經(jīng)廢棄
1.16已經(jīng)標(biāo)記io/ioutil
為廢棄,函數(shù)被轉(zhuǎn)移到了os和io這兩個(gè)包里,具體見下表:
ioutil舊函數(shù) |
新函數(shù) |
Discard |
io.Discard |
NopCloser |
io.NopCloser |
ReadAll |
io.ReadAll |
ReadDir |
os.ReadDir |
ReadFile |
os.ReadFile |
WriteFile |
os.WriteFile |
TempDir |
os.MkdirTemp |
TempFile |
os.CreateTemp |
現(xiàn)在開始可以做移植了。
tcp半連接隊(duì)列擴(kuò)容
在Linux kernel 4.1以前,golang設(shè)置tcp的listen隊(duì)列的長(zhǎng)度是從/proc/sys/net/core/somaxconn獲取的,通常為4096。
而在4.1以后golang會(huì)直接設(shè)置半連接隊(duì)列的長(zhǎng)度為2^32 - 1
也就是4294967295。
更大的半連接隊(duì)列意味著可以同時(shí)處理更多的新加入請(qǐng)求,而且不用再讀取配置文件性能也會(huì)略微提升。
重大更新io/fs
1.16除了支持嵌入靜態(tài)資源外,最大的變化就是引入了io/fs包。
golang認(rèn)為文件的io操作是依賴于文件系統(tǒng)(filesystem,fs)的,所以決定模仿Linux的vfs做一套基于fs的io接口。
這樣做的目的有三個(gè):
- os包應(yīng)該專注于和系統(tǒng)交互而不是包含一部分io接口
- io包和os包分別包含了io接口的一部分,導(dǎo)致互相依賴職責(zé)不清晰
- 可以把有關(guān)聯(lián)的一部分文件或者數(shù)據(jù)組成虛擬文件系統(tǒng),供通用接口處理提升程序的可擴(kuò)展性,比如zip打包的文件
所以io/fs誕生了。
fs包中主要包含了下面幾種數(shù)據(jù)類型(都是接口類型):
|
名稱 |
作用 |
FS |
文件系統(tǒng)的抽象,有一個(gè)Open方法用來從FS打開獲取文件數(shù)據(jù) |
DirEntry |
描述目錄項(xiàng)目(包含目錄自身)的數(shù)據(jù)結(jié)構(gòu) |
File |
描述文件數(shù)據(jù)的結(jié)構(gòu),包含Stat,Read,Close方法 |
ReadDirFile |
在File的基礎(chǔ)上支持ReadDir,可以代表目錄自身 |
FileMode |
描述文件類型,比如是通常文件還是套接字或者是管道 |
FileInfo |
文件的元數(shù)據(jù),例如創(chuàng)建時(shí)間等 |
其中有一些接口和os包中的同名,實(shí)際上是os包引入fs包后起的別名。
對(duì)于FS,還有以下的擴(kuò)展,以便增量描述文件系統(tǒng)允許的操作:
|
名稱 |
作用 |
GlobFS |
增加Glob方法,可以用通配符查找文件 |
ReadDirFS |
增加ReadDir方法,可以遍歷目錄 |
ReadFileFS |
增加ReadFile方法,可以用文件名讀取文件所有內(nèi)容 |
StatFS |
增加Stat方法,可以獲得文件/目錄的元信息 |
SubFS |
增加Sub方法,Sub方法接受一個(gè)文件/目錄的名字,從這個(gè)名字作為根目錄返回一個(gè)新的文件系統(tǒng)對(duì)象 |
fs包還提供了諸如Glob,WalkDir等傳統(tǒng)的文件操作接口。
fs的主要威力在于處理zip、tar文件,以及http的文件接口時(shí)可以大幅簡(jiǎn)化代碼。而且新的embed
靜態(tài)資源嵌入也是依賴fs實(shí)現(xiàn)的。
因?yàn)橹皇撬儆[的緣故,無法詳盡介紹io/fs包,你可以參考golang的文檔或這篇文章做進(jìn)一步了解。
其他改進(jìn)
其他的改進(jìn)包括Unicode更新到了13.0、新增加了runtime/metrics包已提供更好更規(guī)范的運(yùn)行時(shí)信息等。
同時(shí)1.16優(yōu)化了鏈接器,現(xiàn)在它在linux/amd64上比1.15快了20-25%,內(nèi)存占用減少了5-15%。
在Windows上已經(jīng)全面支持了地址空間布局隨機(jī)化(ASLR),此前不支持將golang編譯為dll時(shí)啟用ASLR。
本次更新中語言本身沒有什么變化。
更多信息可以查看golang1.16 release notes
到此這篇關(guān)于golang1.16新特性速覽的文章就介紹到這了,更多相關(guān)golang1.16新特性速覽內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
您可能感興趣的文章:- 分析MongoDB和MySQL各自的關(guān)鍵特性、差別和優(yōu)勢(shì)
- go特性之?dāng)?shù)組與切片的問題
- django 中QuerySet特性功能詳解
- MongoDB的基本特性與內(nèi)部構(gòu)造的講解
- Django 2.0版本的新特性搶先看!
- golang中defer的關(guān)鍵特性示例詳解
- Go語言區(qū)別于其他語言的特性