首页 > 解决方案 > 在 Golang 中构建动态(条件)WHERE SQL 查询

问题描述

我正在使用 golang、go_reform、PostgreSQL。我想做的是一个 REST 搜索实用程序,在我遇到条件搜索查询之前一切都很好。这里的“条件”意味着我在一个表中有 10 列要搜索,并且可能有很多组合,所以我不能单独处理它们。我需要的是一个查询构建器,但我不知道如何在 Go 中实现它。现在我有这样的想法,但它似乎不是很有效

type Query struct {
    Id               *int64
    FirstName        *string
    MiddleName       *string
    LastName         *string
    AreaId           *int64
    Birthday         *time.Time
}

func (table *Query) Find() (*User) {
    if table.Id != nil {
        idstr := fmt.Sprintf("WHERE Id = %d AND ", table.Id)
    }
    else idstr := "WHERE "
    }
    if table.FirstName != "" {
        firststr := fmt.Sprintf("FirstName = %s AND", table.FirstName)
    }
    else firststr := ""
}//and so on

这感觉真的很尴尬,所以我想知道是否有更好的方法来确定到达的字段Find()并基于此构建 SQL 查询。(实际上它以 JSON 形式出现并绑定到Querystruct,所以也许有一种没有 struct 的方法)。也可能有 SQL 解决方法,但我认为在没有所有可能列的情况下构建查询会更有效。

编辑:顺便说一下,使我的谷歌搜索查询更准确,我发现了一堆与我的问题相关的东西,可能我现在会尝试使用它。对于那些也有兴趣的人: 老去游乐场示例

对 MySQL 数据库进行动态 SQL 查询

gorp 包(片段的东西听起来很有希望)

标签: goormprintf

解决方案


所以,我找到了解决方案。非常感谢Cerise Limón,他的代码非常适合我。

我最终得到的解决方案

控制器

func Find(c echo.Context) (err error) {
model := &models.Query{}
if err = c.Bind(model); err != nil {
    return c.JSON(http.StatusInternalServerError, u.Message(false, "Bad request"))
}
resp := model.Find()
return c.JSON(http.StatusOK, resp)

模型

type Query map[string]interface{}

func (model Query) Find() (Query) {
    var values []interface{}
    var where []string
    for k, v := range model {
        values = append(values, v)
        //MySQL Way: where = append(where, fmt.Sprintf("%s = ?", k))
        where = append(where, fmt.Sprintf(`"%s" = %s`,k, "$" + strconv.Itoa(len(values))))
    }
    string := ("SELECT name FROM users WHERE " + strings.Join(where, " AND "))
    //for testing purposes i didn't ran actual query, just print it in the console and returned JSON back
    fmt.Println(string)
    return model

}

更新:对于 PostgreSQL 用户(感谢 @mkopriva 和他的游乐场示例),我可以让这个占位符在 PostgreSQL 上正常工作


推荐阅读