JSON是一種發(fā)送和接收格式化信息的標準。JSON不是唯一的標準,XML、ASN.1 和 Google 的 Protocol Buffer 都是相似的標準。Go通過標準庫 encoding/json、encoding/xml、encoding/asn1 和其他的庫對這些格式的編碼和解碼提供了非常好的支持,這些庫都擁有相同的API。
我們提供的服務有:成都網(wǎng)站制作、成都網(wǎng)站設計、微信公眾號開發(fā)、網(wǎng)站優(yōu)化、網(wǎng)站認證、東乃ssl等。為近千家企事業(yè)單位解決了網(wǎng)站和推廣的問題。提供周到的售前咨詢和貼心的售后服務,是有科學管理、有技術的東乃網(wǎng)站制作公司
首先定義一組數(shù)據(jù):
type Movie struct {
Title string
Year int `json:"released"`
Color bool `json:"color,omitempty"`
Actors []string
}
var movies = []Movie{
{Title: "Casablanca", Year: 1942, Color: false,
Actors: []string{"Humphrey Bogart", "Ingrid Bergman"}},
{Title: "Cool Hand Luke", Year: 1967, Color: true,
Actors: []string{"Paul Newman"}},
{Title: "Bullitt", Year: 1968, Color: true,
Actors: []string{"Steve McQueen", "Jacqueline Bisset"}},
}
然后通過 json.Marshal 進行編碼:
data, err := json.Marshal(movies)
if err != nil {
log.Fatalf("JSON Marshal failed: %s", err)
}
fmt.Printf("%s\n", data)
/* 執(zhí)行結果
[{"Title":"Casablanca","released":1942,"Actors":["Humphrey Bogart","Ingrid Bergman"]},{"Title":"Cool Hand Luke","released":1967,"color":true,"Actors":["Paul Newman"]},{"Title":"Bullitt","released":1968,"color":true,"Actors":["Steve McQueen","Jacqueline Bisset"]}]
*/
這種緊湊的表示方法適合傳輸,但是不方便閱讀。有一個 json.MarshalIndent 的變體可以輸出整齊格式化過的結果。多傳2個參數(shù),第一個是定義每行輸出的前綴字符串,第二個是定義縮進的字符串:
data, err := json.MarshalIndent(movies, "", " ")
if err != nil {
log.Fatalf("JSON Marshal failed: %s", err)
}
fmt.Printf("%s\n", data)
/* 執(zhí)行結果
[
{
"Title": "Casablanca",
"released": 1942,
"Actors": [
"Humphrey Bogart",
"Ingrid Bergman"
]
},
{
"Title": "Cool Hand Luke",
"released": 1967,
"color": true,
"Actors": [
"Paul Newman"
]
},
{
"Title": "Bullitt",
"released": 1968,
"color": true,
"Actors": [
"Steve McQueen",
"Jacqueline Bisset"
]
}
]
*/
只有可導出的成員可以轉換為JSON字段,上面的例子中用的都是大寫。
成員標簽(field tag),是結構體成員的編譯期間關聯(lián)的一些元素信息。標簽值的第一部分指定了Go結構體成員對應的JSON中字段的名字。
另外,Color標簽還有一個額外的選項 omitempty,它表示如果這個成員的值是零值或者為空,則不輸出這個成員到JSON中。所以Title為"Casablanca"的JSON里沒有color。
反序列化操作將JSON字符串解碼為Go數(shù)據(jù)結構。這個是由 json.Unmarshal 實現(xiàn)的。
var titles []struct{ Title string }
if err := json.Unmarshal(data, &titles); err != nil {
log.Fatalf("JSON unmarshaling failed: %s", err)
}
fmt.Println(titles)
/* 執(zhí)行結果
[{Casablanca} {Cool Hand Luke} {Bullitt}]
*/
這里接收數(shù)據(jù)時定義的結構體只有一個Title字段,這樣當函數(shù) Unmarshal 調(diào)用完成后,將填充結構體切片中的 Title 值,而JSON中其他的字段就丟棄了。
很多的 Web 服務器都提供 JSON 接口,通過發(fā)送HTTP請求來獲取想要得到的JSON信息。下面通過查詢Github提供的 issue 跟蹤接口來演示一下。
首先,定義好類型,順便還有常量:
// ch5/github/github.go
// https://api.github.com/ 提供了豐富的接口
// 提供查詢GitHub的issue接口的API
// GitHub上有詳細的API使用說明:https://developer.github.com/v3/search/#search-issues-and-pull-requests
package github
import "time"
const IssuesURL = "https://api.github.com/search/issues"
type IssuesSearchResult struct {
TotalCount int `json:"total_count"`
Items []*Issue
}
type Issue struct {
Number int
HTMLURL string `json:"html_url"`
Title string
State string
User *User
CreateAt time.Time `json:"created_at"`
Body string // Markdown 格式
}
type User struct {
Login string
HTMLURL string `json:"html_url"`
}
關于字段名稱,即使對應的JSON字段的名稱都是小寫的,但是結構體中的字段必須首字母大寫(不可導出的字段也無法把JSON數(shù)據(jù)導入)。這種情況很普遍,這里可以偷個懶。在 Unmarshal 階段,JSON字段的名稱關聯(lián)到Go結構體成員的名稱是忽略大小寫的,這里也不需要考慮序列化的問題,所以很多地方都不需要寫成員標簽。不過,小寫的變量在需要分詞的時候,可能會使用下劃線分割,這種情況下,還是要用一下成員標簽的。
這里也是選擇性地對JSON中的字段進行解碼,因為相對于這里演示的內(nèi)容,GitHub的查詢返回的信息是相當多的。
函數(shù) SearchIssues 發(fā)送HTTP請求并將返回的JSON字符串進行解析。
關于Get請求的參數(shù),參數(shù)中可能會出現(xiàn)URL格式里的特殊字符,比如 ?、&。因此要使用 url.QueryEscape 函數(shù)進行轉義。
// ch5/github/search.go
package github
import (
"encoding/json"
"fmt"
"net/http"
"net/url"
"strings"
)
// 查詢GitHub的issue接口
func SearchIssues(terms []string) (*IssuesSearchResult, error) {
q := url.QueryEscape(strings.Join(terms, " "))
resp, err := http.Get(IssuesURL + "?q=" + q)
if err != nil {
return nil, err
}
defer resp.Body.Close()
if resp.StatusCode != http.StatusOK {
return nil, fmt.Errorf("search query failed: %s", resp.Status)
}
var result IssuesSearchResult
if err := json.NewDecoder(resp.Body).Decode(&result); err != nil {
return nil, err
}
return &result, nil
}
流式解碼噐
之前是使用 json.Unmarshal 進行解碼,而這里使用流式解碼噐。它可以依次從字節(jié)流中解碼出多個JSON實體,不過這里沒有用到該功能。另外還有對應的 json.Encoder 的流式編碼器。
調(diào)用 Decode 方法后,就完成了對變量 result 的填充。
最后就是將 result 中的內(nèi)容進行格式化輸出,這里用了固定寬度的方法將結果輸出為類似表格的形式:
// ch5/issues/main.go
// 將符合條件的issue輸出為一個表格
package main
import (
"fmt"
"gopl/ch5/github"
"log"
"os"
)
func main() {
result, err := github.SearchIssues(os.Args[1:])
if err != nil {
log.Fatal(err)
}
fmt.Printf("%d issue: \n", result.TotalCount)
for _, item := range result.Items {
fmt.Printf("#%-5d %9.9s %.55s\n", item.Number, item.User.Login, item.Title)
}
}
使用命令行參數(shù)指定搜索條件,該命令搜索 Go 項目里的 issue 接口,查找 open 狀態(tài)的列表。由于返回的還是很多,后面的參數(shù)是對內(nèi)容再進行篩選:
PS H:\Go\src\gopl\ch5\issues> go run main.go repo:golang/go is:open json decoder tag
6 issue:
#28143 Carpetsmo proposal: encoding/json: add "readonly" tag
#14750 cyberphon encoding/json: parser ignores the case of member names
#17609 nathanjsw encoding/json: ambiguous fields are marshalled
#22816 ganelon13 encoding/json: include field name in unmarshal error me
#19348 davidlaza cmd/compile: enable mid-stack inlining
#19109 bradfitz proposal: cmd/go: make fuzzing a first class citizen, l
PS H:\Go\src\gopl\ch5\issues>
進行簡單的格式化輸出,使用fmt包就足夠了。但是要實現(xiàn)更復雜的格式化輸出,并且有時候還要求格式和代碼徹底分離。這可以通過 text/templat 包和 html/template 包里的方法來實現(xiàn),通過這兩個包,可以將程序變量的值代入到模板中。
模板,是一個字符串或者文件,它包含一個或者多個兩邊用雙大括號包圍的單元,這稱為操作。大多數(shù)字符串是直接輸出的,但是操作可以引發(fā)其他的行為。
每個操作在模板語言里對應一個表達式,功能包括:
這篇里有表達式的介紹: https://blog.51cto.com/steed/2321827
繼續(xù)使用 GitHub 的 issue 接口返回的數(shù)據(jù),這次使用模板來輸出。一個簡單的字符串模板如下所示:
const templ = `{{.TotalCount}} issues:
{{range .Items}}----------------------------------------
Number: {{.Number}}
User: {{.User.Login}}
Title: {{.Title | printf "%.64s"}}
Age: {{.CreatedAt | daysAgo}} days
{{end}}`
點號(.)表示當前值的標記。最開始的時候表示模板里的參數(shù),也就是 github.IssuesSearchResult。
操作 {{.TotalCount}} 就是 TotalCount 字段的值。
{{range .Items}} 和 {{end}} 操作創(chuàng)建一個循環(huán),這個循環(huán)內(nèi)部的點號(.)表示Items里的每一個元素。
在操作中,管道符(|)會將前一個操作的結果當做下一個操作的輸入,這個和UNIX里的管道類似。{{.Title | printf "%.64s"}}
,這里的第二個操作是printf函數(shù),在包里這個名稱對應的就是fmt.Sprintf,所以會按照fmt.Sprintf函數(shù)返回的樣式輸出。{{.CreatedAt | daysAgo}}
,這里的第二個操作數(shù)是 daysAgo,這是一個自定義的函數(shù),具體如下:
func daysAgo(t time.Time) int {
return int(time.Since(t).Hours() / 24)
}
通過模板輸出結果需要兩個步驟:
解析模板只需要執(zhí)行一次。下面的代碼創(chuàng)建并解析上面定義的文本模板:
report, err := template.New("report").
Funcs(template.FuncMap{"daysAgo": daysAgo}).
Parse(templ)
if err != nil {
log.Fatal(err)
}
這里使用了方法的鏈式調(diào)用。template.New 函數(shù)創(chuàng)建并返回一個新的模板。
Funcs 方法將自定義的 daysAgo 函數(shù)到內(nèi)部的函數(shù)列表中。之前提到的printf實際對應的是fmt.Sprintf,也是在包內(nèi)默認就已經(jīng)在這個函數(shù)列表里了。如果有更多的自定義函數(shù),就多次調(diào)用這個方法添加。
最后就是調(diào)用Parse進行解析。
上面的代碼完成了創(chuàng)建模板,添加內(nèi)部可調(diào)用的 daysAgo 函數(shù),解析(Parse方法),檢查(檢查err是否為空)?,F(xiàn)在就可以調(diào)用report的 Execute 方法,傳入數(shù)據(jù)源(github.IssuesSearchResult,這個需要先調(diào)用github.SearchIssues函數(shù)來獲?。?,并指定輸出目標(使用 os.Stdout):
if err := report.Execute(os.Stdout, result); err != nil {
log.Fatal(err)
}
之前的代碼比較凌亂,下面出完整可運行的代碼:
package main
import (
"log"
"os"
"text/template"
"time"
"gopl/ch5/github"
)
const templ = `{{.TotalCount}} issues:
{{range .Items}}----------------------------------------
Number: {{.Number}}
User: {{.User.Login}}
Title: {{.Title | printf "%.64s"}}
Age: {{.CreatedAt | daysAgo}} days
{{end}}`
// 自定義輸出格式的方法
func daysAgo(t time.Time) int {
return int(time.Since(t).Hours() / 24)
}
func main() {
// 解析模板
report, err := template.New("report").
Funcs(template.FuncMap{"daysAgo": daysAgo}).
Parse(templ)
if err != nil {
log.Fatal(err)
}
// 獲取數(shù)據(jù)
result, err := github.SearchIssues(os.Args[1:])
if err != nil {
log.Fatal(err)
}
// 輸出
if err := report.Execute(os.Stdout, result); err != nil {
log.Fatal(err)
}
}
這個版本還可以改善,下面對解析錯誤的處理進行了改進
由于目標通常是在編譯期間就固定下來的,因此無法解析將會是一個嚴重的bug。上面的版本如果無法解析(去掉個大括號試試),只會以比較溫和的方式報告出來。
這里推薦使用幫助函數(shù) template.Must,模板錯誤會Panic:
package main
import (
"log"
"os"
"text/template"
"time"
"gopl/ch5/github"
)
const templ = `{{.TotalCount}} issues:
{{range .Items}}----------------------------------------
Number: {{.Number}}
User: {{.User.Login}}
Title: {{.Title | printf "%.64s"}}
Age: {{.CreatedAt | daysAgo}} days
{{end}}`
// 自定義輸出格式的方法
func daysAgo(t time.Time) int {
return int(time.Since(t).Hours() / 24)
}
// 使用幫助函數(shù)
var report = template.Must(template.New("issuelist").
Funcs(template.FuncMap{"daysAgo": daysAgo}).
Parse(templ))
func main() {
result, err := github.SearchIssues(os.Args[1:])
if err != nil {
log.Fatal(err)
}
if err := report.Execute(os.Stdout, result); err != nil {
log.Fatal(err)
}
}
和上個版本的區(qū)別就是解析的過程外再包了一層 template.Must 函數(shù)。而效果就是原本解析錯誤是調(diào)用 log.Fatal(err)
來退出,這個調(diào)用也是自己的代碼里指定的。
而現(xiàn)在是調(diào)用 panic(err)
來退出,并且會看到一個更加嚴重的錯誤報告(錯誤信息是一樣的),并且這個也是包內(nèi)部提供的并且推薦的做法。
最后是輸出的結果:
PS H:\Go\src\gopl\ch5\issuesreport> go run main.go repo:golang/go is:open json decoder tag
6 issues:
----------------------------------------
Number: 28143
User: Carpetsmoker
Title: proposal: encoding/json: add "readonly" tag
Age: 135 days
----------------------------------------
Number: 14750
User: cyberphone
Title: encoding/json: parser ignores the case of member names
Age: 1079 days
----------------------------------------
...
接著看 html/template 包。它使用和 text/template 包里一樣的 API 和表達式語法,并且額外地對出現(xiàn)在 HTML、JavaScript、CSS 和 URL 中的字符串進行自動轉義。這樣可以避免在生成 HTML 是引發(fā)一些安全問題。
下面是一個將 issue 輸出為 HTML 表格代碼。由于兩個包里的API是一樣的,所以除了模板本身以外,GO代碼沒有太大的差別:
package main
import (
"fmt"
"log"
"net/http"
"os"
)
import (
"gopl/ch5/github"
"html/template"
)
var issueList = template.Must(template.New("issuelist").Parse(`
{{.TotalCount}} issues
#
State
User
Title
{{range .Items}}
{{.Number}}
{{.State}}
{{.User.Login}}
{{.Title}}
{{end}}
`))
func main() {
result, err := github.SearchIssues(os.Args[1:])
if err != nil {
log.Fatal(err)
}
fmt.Println("http://localhost:8000")
handler := func(w http.ResponseWriter, r *http.Request) {
showIssue(w, result)
}
http.HandleFunc("/", handler)
log.Fatal(http.ListenAndServe("localhost:8000", nil))
}
func showIssue(w http.ResponseWriter, result *github.IssuesSearchResult) {
if err := issueList.Execute(w, result); err != nil {
log.Fatal(err)
}
}
通過模板的操作導入的字符串,默認都會按照原樣顯示出來。就是會把HTML的特殊字符自動進行轉義,效果就是無法通過模板導入的內(nèi)容生成html標簽。
如果就是需要通過模板的操作再導入一些HTML的內(nèi)容,就需要使用 template.HTML 類型。使用 template.HTML 類型后,可以避免模板自動轉義受信任的 HTML 數(shù)據(jù)。同樣的類型還有 template.CSS、template.JS、template.URL 等,具體可以查看源碼。
下面的操作演示了普通的 string 類型和 template.HTML 類型在導入一個 HTML 標簽后顯示效果的差別:
package main
import (
"fmt"
"html/template"
"log"
"net/http"
)
func main() {
const templ = `A: {{.A}}
B: {{.B}}
`
t := template.Must(template.New("escape").Parse(templ))
var data struct {
A string // 不受信任的純文本
B template.HTML // 受信任的HTML
}
data.A = "Hello!"
data.B = "Hello!"
fmt.Println("http://localhost:8000")
handler := func(w http.ResponseWriter, r *http.Request) {
if err := t.Execute(w, data); err != nil {
log.Fatal(err)
}
}
http.HandleFunc("/", handler)
log.Fatal(http.ListenAndServe("localhost:8000", nil))
}