《GO程序設計語言》設計中案例,僅作爲筆記進行收藏。此案例將HTTP請求參數解析成對應的匿名結構體,並使用反射來獲取字段標籤。
- params 解析函數
package params
import (
"fmt"
"net/http"
"reflect"
"strconv"
"strings"
)
// Unpack 從 HTTP 請求 req 的參數中提取數據填充到 ptr 指向結構體的各個字段
func Unpack(req *http.Request, ptr interface{}) error {
if err := req.ParseForm(); err != nil {
return err
}
// 創建字段映射表,鍵爲有效名稱
fields := make(map[string]reflect.Value)
v := reflect.ValueOf(ptr).Elem()
for i := 0; i < v.NumField(); i++ {
fieldInfo := v.Type().Field(i)
tag := fieldInfo.Tag
name := tag.Get("http")
if name == "" {
name = strings.ToLower(fieldInfo.Name)
}
fields[name] = v.Field(i)
}
// 對請求中的每個參數更新結構體中對應的字段
for name, values := range req.Form {
f := fields[name]
if !f.IsValid() {
continue // 忽略不能識別的 HTTP 參數
}
for _, value := range values {
if f.Kind() == reflect.Slice {
elem := reflect.New(f.Type().Elem()).Elem()
if err := populate(elem, value); err != nil {
return fmt.Errorf("%s: %v", name, err)
}
f.Set(reflect.Append(f, elem))
} else {
if err := populate(f, value); err != nil {
return fmt.Errorf("%s: %v", name, err)
}
}
}
}
return nil
}
func populate(v reflect.Value, value string) error {
switch v.Kind() {
case reflect.String:
v.SetString(value)
case reflect.Int:
i, err := strconv.ParseInt(value, 10, 64)
if err != nil {
return err
}
v.SetInt(i)
case reflect.Bool:
b, err := strconv.ParseBool(value)
if err != nil {
return err
}
v.SetBool(b)
default:
return fmt.Errorf("unsupported kind %s", v.Type())
}
return nil
}
- search 請求
package main
import (
"fmt"
"log"
"net/http"
"main/params"
)
func search(resp http.ResponseWriter, req *http.Request) {
var data struct {
Labels []string `http:"l"`
MaxResults int `http:"max"`
Exact bool `http:"x"`
}
// 設置默認值
data.MaxResults = 10
if err := params.Unpack(req, &data); err != nil {
http.Error(resp, err.Error(), http.StatusBadRequest)
return
}
fmt.Fprintf(resp, "Search:%+v\n", data)
}
func main() {
http.HandleFunc("/search", search)
log.Fatal(http.ListenAndServe(":8080", nil))
}