一、结构体基础
结构体 (struct) 将多个不同类型的字段集中组成一种复合类型,按声明时的字段顺序初始化。
type user struct {
name string
age byte
}
user := user {"Tom", 2}
定义匿名结构体时没有 type 关键字,与其他定义类型的变量一样,如果在函数外部需在结构体变量前加上 var 关键字,在函数内部可省略 var 关键字。
// 在函数外部定义匿名结构体并赋值给
var config struct {
APIKey string
OAuthConfig oauth.Config
}// 定义并初始化并赋值给
datadata := struct {
Title string
Users []*User
}{
title,
users
}
二、匿名结构体使用场景
1、组织全局变量
属于同一类的全局变量可通过匿名结构体组织在一起。
var config struct {
APIKey string
OAuthConfig oauth.Config
}
config.APIKey = "BADC0C0A"
2、数据模版
可在后端把数据组织成前端需要的格式传给渲染模版
package mainimport ( "html/template"
"net/http"
"strings")type Paste struct {
Expiration string
Content []byte
UUID string}func pasteHandler(w http.ResponseWriter, r *http.Request) {
paste_id := strings.TrimPrefix(r.URL.Path, "/paste")
paste := &Paste{UUID: paste_id}
keep_alive := false
burn_after_reading := false
data := struct {
Paste *Paste
KeepAlive bool
BurnAfterReading bool
} {
paste,
keep_alive,
burn_after_reading,
}
t, _ := template.ParseFiles("templates/paste.html")
t.Execute(w, data)
}
匿名函授定义并初始化之后赋值给 data 变量,除了把 Paste 结构体对象的值传给前端之外,还额外添加了必要的字段。
写过前后端的同学应该知道,前端有时需要后端返回一个标志变量作为判断条件是否显示某一块内容。
Expiration: {{ .Paste.Expiration }}
UUID: {{ .Paste.UUID}}
{{ if .BurnAfterReading }}
BurnAfterReading: True{{ else }}
BurnAfterReading: False{{ end }}
3、测试案例数据
在写测试代码时,经常用到匿名结构体生成用例的输入输出,为了覆盖各个测试维度,通常结合切片使用,构成了测试样例尽可能地覆盖所有可能发生情况。
var indexRuneTests = []struct {
s string
rune rune out int}{
{"a A x", 'A', 2},
{"some_text=some_value", '=', 9},
{"a", 'a', 3},
{"ab", '', 4},
}
4、嵌入式锁 (Embedded lock)
var hits struct {
sync.Mutex
n int}
hits.Lock()
hits.n++
hits.Unlock()
三、案例
手机拥有屏幕、电池、指纹识别等信息,将这些信息填充为 JSON 格式的数据。如果需要选择性地分离 JSON 中的数据则较为麻烦。Golang中的匿名结构体可以方便地完成这个操作。
首先给出完整的代码,然后再讲解每个部分。
package main
import (
"encoding/json"
"fmt"
)
//定义手机屏幕
type Screen01 struct {
Size float64 //屏幕尺寸
ResX, ResY int //屏幕分辨率 水平 垂直
}
//定义电池容量
type Battery struct {
Capacity string
}
//返回json数据
func getJsonData() []byte {
//tempData 接收匿名结构体(匿名结构体使得数据的结构更加灵活)
tempData := struct {
Screen01
Battery
HashTouchId bool // 是否有指纹识别
}{
Screen01: Screen01{Size: 12, ResX: 36, ResY: 36},
Battery: Battery{"6000毫安"},
HashTouchId: true,
}
jsonData, _ := json.Marshal(tempData) //将数据转换为json
return jsonData
}
func main() {
jsonData := getJsonData() //获取json数据
fmt.Println(jsonData)
fmt.Println("=========解析(分离)出的数据是===========")
//自定义匿名结构体,解析(分离)全部数据
allData := struct {
Screen01
Battery
HashTouchId bool
}{}
json.Unmarshal(jsonData, &allData)
fmt.Println("解析(分离)全部结构为:", allData)
//自定义匿名结构体,通过json数据,解析(分离)对应的结构(可以是部分结构)
screenBattery := struct {
Screen01
Battery
}{}
json.Unmarshal(jsonData, &screenBattery) //注意:此处只能为结构体指针(一般参数为interface{},都采用地址引用(即地址传递))
fmt.Println("解析(分离)部分结构:", screenBattery)
//自定义匿名结构体,解析(分离)部分结构
batteryTouch := struct {
Battery
isTouch bool
}{}
json.Unmarshal(jsonData, &batteryTouch)
fmt.Println("解析(分离)部分结构:", batteryTouch)
//自定义匿名结构体,解析(分离)部分不存在的结构
temp1 := struct {
Battery
Detail struct {
Name string
Price uint16
}
}{}
json.Unmarshal(jsonData, &temp1)
fmt.Println("解析(分离)部分不存在的结构", temp1)
//自定义匿名结构体,解析(分离)完全不存在的结构
temp2 := struct {
User string
Price uint16
}{}
json.Unmarshal(jsonData, &temp2)
fmt.Println("解析(分离)完全不存在的结构:", temp2)
}
注:匿名结构体可以组合不同类型的数据,使得处理数据变得更为灵活。尤其是在一些需要将多个变量、类型数据组合应用的场景,匿名结构体是一个不错的选择。
json.Unmarshal(data []byte, v interface{})
//注意:参数 v 只能为结构体指针(一般参数为interface{},都采用地址引用(即地址传递))
原文 https://blog.csdn.net/weixin_42117918/article/details/90448756