Go1.16新特性embed打包靜態(tài)資源文件實(shí)現(xiàn)
背景
相信有一部分人喜愛 GO 的初衷大概是:跨平臺(tái)靜態(tài)編譯,如果在沒用通過 CGO 引用其他庫的話,一般編譯出來的可執(zhí)行二進(jìn)制文件部署起來非常方便,但人們?cè)趯?shí)際中發(fā)現(xiàn),使用 Go 語言開發(fā)的后端 WEB 程序存在 HTML 模版、圖片、JS、CSS、JSON 等靜態(tài)資源,部署時(shí)需要把這些靜態(tài)資源與二進(jìn)制程序一起上傳到服務(wù)器部署,在現(xiàn)今遍地花容器的今天,為了簡化部署流程,能不能更進(jìn)一步的把這些靜態(tài)文件與二進(jìn)制程序一起打包起來部署,那樣豈不是更方便?對(duì)運(yùn)維來說打包一體化帶來的好處就是運(yùn)維復(fù)雜度的降低,對(duì)技術(shù)團(tuán)隊(duì)來說打包一體化還可以保證程序完整性可控性。
因此,GO 社區(qū)發(fā)起了一個(gè)期望 Go 編譯器支持嵌入靜態(tài)文件的提案issue#35950?,F(xiàn)在,這個(gè)功能將隨著 1.16 版本一起發(fā)布,目前最新的版本是 Go 1.16 RC1 預(yù)覽版。
好了,接下來我們?cè)敿?xì)介紹 go embed 的各個(gè)功能。
embed 嵌入
└── cmd 測試目錄
├── assets 靜態(tài)資源目錄
│?? ├── .idea.txt
│?? ├── golang.txt
│?? └── hello.txt
└── main.go 測試go源文件
字符串、字節(jié)切片、文件嵌入
package main
import (
"embed"
_ "embed"
"fmt"
)
//go:embed指令用來嵌入,必須緊跟著嵌入后的變量名
//只支持嵌入為string, byte slice和embed.FS三種類型,這三種類型的別名(alias)和命名類型(如type S string)都不可以
//以字符串形式嵌入 assets/hello.txt
//go:embed assets/hello.txt
var s string
//文件的內(nèi)容嵌入為slice of byte,也就是一個(gè)字節(jié)數(shù)組
//go:embed assets/hello.txt
var b []byte
//嵌入為一個(gè)文件系統(tǒng) 新的文件系統(tǒng)FS
//go:embed assets/hello.txt
//go:embed assets/golang.txt
var f embed.FS
func main() {
fmt.Println("embed string.", s)
fmt.Println("embed byte.", string(b))
data, _ := f.ReadFile("assets/hello.txt")
fmt.Println("embed fs.", string(data))
data, _ = f.ReadFile("assets/golang.txt")
fmt.Println("embed fs.", string(data))
}
編譯運(yùn)行后輸出:
embed string. hello golang!
embed byte. hello golang!
embed fs. hello golang!
embed fs. hello!
從上面的代碼可以看出,embed 支持嵌入為 string,byte slice 和 embed.FS 這三種類型,另外也不允許從這些類型中派生哦。
嵌入文件
對(duì)于 FS 類型的嵌入,也可以支持一個(gè)變量嵌入多個(gè)文件。
//go:embed assets/hello.txt //go:embed assets/golang.txt var f embed.FS
當(dāng)然也支持,兩個(gè)變量嵌入一個(gè)文件。雖然兩個(gè)變量嵌入了同一個(gè)文件,但它們?cè)诰幾g的時(shí)候會(huì)獨(dú)立分配,彼此之間并不會(huì)互相影響。
嵌入文件夾
FS 類型的嵌入支持文件夾.分隔符采用正斜杠/,即使是 windows 系統(tǒng)也采用這個(gè)模式。
//go:embed assets
var f embed.FS
func main() {
data, _ := f.ReadFile("assets/hello.txt")
fmt.Println(string(data))
}
嵌入匹配
go:embed 指令中可以只寫文件夾名,此文件夾中除了.和_開頭的文件和文件夾都會(huì)被嵌入,并且子文件夾也會(huì)被遞歸的嵌入,形成一個(gè)此文件夾的文件系統(tǒng)。
如果想嵌入.和_開頭的文件和文件夾, 比如.hello.txt 文件,那么就需要使用*,比如 go:embed assets/*。
不具有遞歸性,所以子文件夾下的.和_不會(huì)被嵌入,除非你在專門使用子文件夾的進(jìn)行嵌入:
├── assets
│?? ├── .idea.txt
│?? ├── golang.txt
│?? └── hello.txt
└── main.go
package main
import (
"embed"
_ "embed"
"fmt"
)
//go:embed assets/*
var f embed.FS
func main() {
data, _ := f.ReadFile("assets/.idea.txt")
fmt.Println(string(data))
}
FS 文件系統(tǒng)
embed.FS 實(shí)現(xiàn)了 io/fs.FS 接口,它可以打開一個(gè)文件,返回 fs.File:
package main
import (
"embed"
_ "embed"
"fmt"
)
//go:embed assets
var f embed.FS
func main() {
helloFile, _ := f.Open("assets/hello.txt")
stat, _ := helloFile.Stat()
fmt.Println(stat.Name(), stat.Size())
}
它還提供了 ReadFileh 和 ReadDir 功能,遍歷一個(gè)文件下的文件和文件夾信息:
package main
import (
"embed"
_ "embed"
"fmt"
)
//go:embed assets
var f embed.FS
func main() {
dirEntries, _ := f.ReadDir("assets")
for _, de := range dirEntries {
fmt.Println(de.Name(), de.IsDir())
}
}
因?yàn)樗鼘?shí)現(xiàn)了 io/fs.FS 接口,所以可以返回它的子文件夾作為新的文件系統(tǒng):
package main
import (
"embed"
_ "embed"
"fmt"
"io/fs"
"io/ioutil"
)
//go:embed assets
var f embed.FS
func main() {
as, _ := fs.Sub(f, "assets")
hi, _ := as.Open("hello.txt")
data, _ := ioutil.ReadAll(hi)
fmt.Println(string(data))
}
總結(jié):
- 對(duì)于單個(gè)的文件,支持嵌入為字符串和 byte slice
- 對(duì)于多個(gè)文件和文件夾,支持嵌入為新的文件系統(tǒng) FS
- go:embed 指令用來嵌入,必須緊跟著嵌入后的變量名
- 只支持嵌入為 string, byte slice 和 embed.FS 三種類型,類型派生也不可以。
以上就是Go 1.16新特性embed打包靜態(tài)資源文件實(shí)現(xiàn)的詳細(xì)內(nèi)容,更多關(guān)于Go embed打包靜態(tài)資源的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
GoFrame框架gredis優(yōu)雅的取值和類型轉(zhuǎn)換
這篇文章主要為大家介紹了GoFrame框架gredis優(yōu)雅的取值和類型轉(zhuǎn)換,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-06-06
Golang執(zhí)行g(shù)o get私有庫提示"410 Gone" 的問題及解決辦法
這篇文章主要介紹了Golang執(zhí)行g(shù)o get私有庫提示”410 Gone“ 解決辦法,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-02-02
Golang基于Vault實(shí)現(xiàn)敏感數(shù)據(jù)加解密
數(shù)據(jù)加密是主要的數(shù)據(jù)安全防護(hù)技術(shù)之一,敏感數(shù)據(jù)應(yīng)該加密存儲(chǔ)在數(shù)據(jù)庫中,降低泄露風(fēng)險(xiǎn),本文將介紹一下利用Vault實(shí)現(xiàn)敏感數(shù)據(jù)加解密的方法,需要的可以參考一下2023-07-07

