在Go語言中,接口是一種非常強(qiáng)大和靈活的類型。然而,當(dāng)我們嘗試讀取一個接口值中的結(jié)構(gòu)地址時,可能會遇到一些困惑。那么,如何讀取具有結(jié)構(gòu)地址的接口值呢?在本文中,我們將為您解答這個問題,并提供一些實(shí)用的技巧和示例代碼。無論您是初學(xué)者還是有經(jīng)驗(yàn)的Go開發(fā)者,本文都將對您有所幫助。讓我們一起來看看吧!
問題內(nèi)容
我有數(shù)據(jù)類型為 interface{} 的變量,并將結(jié)構(gòu)體的地址傳遞給它。現(xiàn)在我無法從界面讀取字段
代碼如下:
type UserData struct {
UserID string `json:"user_id"`
UserName string `json:"user_name"`
}
type Result struct {
Status string `json:"status"`
Data interface{} `json:"data"`
}
var res Result
res.Data = &UserData
json.Unmarshal([]byte(`{"status": "success", "data": {"user_id":15,"user_name":"abc"}}`), &res)
fmt.Println(res.Data) //working fine
fmt.Println(res.Data.UserName) //getting error: type interface{} has no field or method UserName
登錄后復(fù)制
如果我使用 res.data.username 出現(xiàn)錯誤
如何從接口讀取結(jié)構(gòu)體字段?
解決方法
將其與 golang 為什么 go 結(jié)構(gòu)中不存在的字段在將所述結(jié)構(gòu)編組到 json 后仍然存在。起初我以為它們是一樣的。但事實(shí)證明并非如此。
對于這個問題,res.data 擁有 *userdata 類型的值。所以一個簡單的類型斷言就可以了。
package main
import (
"encoding/json"
"fmt"
)
type userdata struct {
userid string `json:"user_id"`
username string `json:"user_name"`
}
type result struct {
status string `json:"status"`
data interface{} `json:"data"`
}
func main() {
var res result
res.data = &userdata{}
json.unmarshal([]byte(`{"status": "success", "data": {"user_id":15,"user_name":"abc"}}`), &res)
fmt.println(res.data)
fmt.println(res.data.(*userdata).username)
}
登錄后復(fù)制
以下演示是 @mkopriva 的兩個演示的合并,它顯示了差異:
package main
import (
"encoding/json"
"fmt"
"log"
)
type dbbatch struct {
fieldtokeep string `json:"field_to_keep"`
fieldtokeep2 string `json:"field_to_keep2"`
}
func main() {
jsonbatch := `{"field_to_keep":"xxxxx","field_to_keep2":"26400527","field_to_delete":"whynotdeleted"}`
var i interface{} = dbbatch{}
fmt.printf("%t\n", i) // type is dbbatch
if err := json.unmarshal([]byte(jsonbatch), &i); err != nil {
log.println(err)
}
fmt.printf("%t\n", i) // type is not dbbatch anymore, instead it's map[string]any
i = &dbbatch{}
fmt.printf("%t\n", i) // type is *dbbatch
if err := json.unmarshal([]byte(jsonbatch), &i); err != nil {
log.println(err)
}
fmt.printf("%t\n", i) // type is *dbbatch
}
登錄后復(fù)制
輸出為:
main.DBBatch
map[string]interface {}
*main.DBBatch
*main.DBBatch
登錄后復(fù)制






