Go 语言反射和范型在 API 服务中的应用

Go reflect
  1. 为何需要使用 reflect 获取:减少重复代码

1. API 接口中抽取参数的逻辑大量重复

  API 接口自然是要获取传过来的数据,不同接口要获取的数据自然也不一样,如果不做特殊处理,必然是每个接口都有一堆功能重复的从 request 里获取参数的代码。

2. API 框架提供的抽取参数的方式并不满足需求

  当然 API 框架会提供这些功能,不过有些情况不能满足需求,比如gin-gonic,提供了将将 request 转为对应结构体的函数,但存在两个问题,第一个问题是参数区分大小写,我觉得应该实现大小写的通配,这样健壮性更高;第二是结构体直接对应数据库表结构,部分数据是不应该从接口请求中读取的,比如创建时间和删除标志,全转换的方式就很有问题。
  不过也有可能是因为我对 gin 不熟悉,不知道更好的用法,才自己造轮子,如果大家有更简洁漂亮的写法,请不吝赐教。

3. Golang 强类型语言的限制

  Go 语言是强类型语言,函数间传递参数或者返回值,必须有特定的类型,如果要实现这种范类型的处理相对 Python 等弱类型的语言要困难一些。
  Python 对于 struct 参数没有严格的限制,传什么内容都行,Golang 就没那么友好了,这部分要靠范型来处理。

# struct 是要获得的数据结果,params 是要抽取的参数名称数组,request 是接口的请求结构体。
def ExtractParamFromBody(struct, params, request):
    ...

  还有一点就是要能获取到 struct 结构体中每个参数的类型,并且给其赋值,Golang 提供的 reflect 机制可以很好的完成这项功能。

4. 实例

  以下代码先是建立了数据库连接(请注意,数据的连接需要提前建立好,并按照代码中的用户名、密码、地址、端口和数据库名称建立,不然代码无法运行成功);之后在数据库中建立了一个叫 User 的表;之后有一个创建用户的接口 "POST /users",对应的函数为 CreateUser。
  ExtractParamFromBody 是通用的参数抽取函数,不光是 User 类型,interface{} 是 Golang 中范型,可以对应任何结构体。

package main

import (
    "fmt"
    "reflect"
    "strconv"
    "strings"

    "github.com/gin-gonic/gin"
    "github.com/jinzhu/gorm"
)

type User struct {
    ID              uint   `json:"id" gorm:"PRIMARY_KEY;AUTO_INCREMENT"`
    Name            string `json:"name" gorm:"INDEX:name;UNIQUE;NOT NULL;type:varchar(100)"`
    Password        string `json:"password" gorm:"NOT NULL"`
    Mobile          string `json:"mobile"`
    Email           string `json:"email"`
    Role_id         uint   `json:"role_id"`
    Create_Time     uint   `json:"create_time"`
    Login_Time      uint   `json:"login_time"`
    Last_Login_Time uint   `json:"last_login_time`
    Login_Count     uint   `json:"login_count"`
    Deleted         bool   `json:"deleted" gorm:"DEFAULT:0"`
}

var db *gorm.DB
var err error

func ExtractParamFromBody(s interface{}, params []string, c *gin.Context) {
    var typ reflect.Type
    var val reflect.Value

    ptyp := reflect.TypeOf(s)

    if ptyp.Kind() == reflect.Ptr {
        val = reflect.ValueOf(s).Elem()
        typ = reflect.TypeOf(s).Elem()
    } else {
        val = reflect.ValueOf(s)
    }

    for _, param := range params {
        ret := c.PostForm(param)
        if ret != "" {
            for i := 0; i < typ.NumField(); i++ {
                if strings.ToLower(typ.Field(i).Name) == param {
                    if val.Field(i).Kind() == reflect.String && val.CanSet() {
                        val.Field(i).SetString(ret)
                    } else if val.Field(i).Kind() == reflect.Uint && val.CanSet() {
                        ret_int, _ := strconv.Atoi(ret)
                        val.Field(i).SetUint(uint64(ret_int))
                    }
                }
            }
        }
    }
}

func InitMysql() *gorm.DB {
    mysql_connection_string := fmt.Sprintf("%s:%s@tcp(%s:%s)/%s?charset=utf8&parseTime=True&loc=Local",
        "root",
        "mysql",
        "127.0.0.1",
        "3306",
        "test_db")
    db, err = gorm.Open("mysql", mysql_connection_string)

    if err != nil {
        log.Logger.Critical("Fail to connect MySQL: %s. Exit.", err)
        os.Exit(5)
    }

    db.AutoMigrate(&User{})
    return db
}

func CreateUser(c *gin.Context) {
    var user mysql.User

    params := []string{"name", "password", "email", "mobile", "role_id"}
    ExtractParamFromBody(&user, params, c)

    if err := db.Create(&user).Error; err != nil {
        c.JSON(200, gin.H{
            "code":   3,
            "result": "failed",
            "msg":    "Fail to create user",
        })
    } else {
        c.JSON(200, gin.H{
            "code":       0,
            "result":     "success",
            "msg":        "success",
            "resultBean": user,
        })
    }
}

func main() {
    InitMysql()

    r := gin.Default()
    r.POST("/v1/users", CreateUser)
    r.Run(":8080")
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 162,475评论 4 372
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 68,744评论 2 307
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 112,101评论 0 254
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,732评论 0 221
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 53,141评论 3 297
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 41,049评论 1 226
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 32,188评论 2 320
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,965评论 0 213
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,716评论 1 250
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,867评论 2 254
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,341评论 1 265
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,663评论 3 263
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,376评论 3 244
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,200评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,990评论 0 201
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 36,179评论 2 285
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,979评论 2 279

推荐阅读更多精彩内容

  • 1、引言 数据库设计过程中表、字段等的命名规范也算是设计规范的一部分,不过设计规范更多的是为了确保数据库设计的合理...
    SnowflakeCloud阅读 40,663评论 0 48
  • 原文链接:https://medium.com/@cgrant/developing-a-simple-crud-...
    devabel阅读 28,274评论 3 52
  • 前言:三毛说,岁月极美,在于它的必然流逝。春花,秋月,夏日,冬雪。又是一年一度的七夕节,楼主也成功的迎来了又一个被...
    北堂野望阅读 557评论 5 8
  • DOM 目前的通用版本是DOM3 1. JS初级主要就两个作用: 1.找元素2.给元素增加/删除class 2. ...
    彭奕泽阅读 96评论 0 1
  • 1. 所有好好喜欢防弹的人不会讨厌南俊的,他照顾成员们也和他们友好地开玩笑,显示了他是多么可靠的队长。音乐和心理方...
    Pancake613阅读 225评论 0 0