0

TLDR如果您尝试运行它,这是一个演示问题的游乐场:https: //play.golang.org/p/myQtUVg1iq

我正在制作一个 REST API,并且有多种类型的资源可以通过 GET 请求检索

GET http://localhost/api/users
GET http://localhost/api/groups

我有一个模型包,它抽象了不同资源的实现方式:

func(m *UserManager) Get() []Users {
    // Internal logic, assume returns correct results
}

func(m *GroupManager) Get() []Groups {
    // Internal logic, assume returns correct results
}

路由文件设置所有路由和处理程序:

users := models.UserManager{}
groups := models.GroupManager{}

func GetUsersHandler (w http.ResponseWriter, r *http.Request) {
    users := users.Get()
    // Implementation details, writing to w as JSON
}

func GetGroupsHandler (w http.ResponseWriter, r *http.Request) {
    groups := groups.Get()
    // Implementation details, writing to w as JSON
}

func registerRoutes(r *mux.Router) {
    r.handleFunc("/api/users", GetUsersHandler).Method("GET")
    r.handleFunc("/api/groups", GetGroupsHandler).Method("GET")
}

我试图通过创建一个接口来使其更通用,然后只需要一个 GetHandler。像这样的东西:

type Getter interface {
    Get() []interface{}
}

func GetHandler(g Getter) {
    return func(w http.ResponseWriter, r *http.Request) {
        results := g.Get()
        // Implementation details, writing to w as JSON
    }
}

func registerRoutes(r *mux.Router) {
    r.handleFunc("/api/users", GetHandler(&users)).Method("GET")
    r.handleFunc("/api/groups", GetHandler(&groups)).Method("GET")
}

这真的很接近工作,唯一的问题是模型的返回类型是特定的对象类型,但接口只使用接口返回类型。有没有办法在不让模型返回的情况下解决这个问题[]interface{}

https://play.golang.org/p/myQtUVg1iq

4

2 回答 2

0

尽量不要像其他 OOP 语言那样处理问题。在 Go 中不能有协变容器,所以要么必须使用空容器,要么必须以interface{}不同的方式构建程序。

如果您的Get方法不同并且您想在接口中对类型进行分组,请使用另一种方法(有时我们甚至有仅用于接口的 noop 方法),或者只是传入usersgroups作为interface{}. 无论如何,您都需要在调用链中的某个点进行类型切换或断言,一旦您知道它是什么类型,您就可以相应地处理它。

没有更多代码很难判断,但在这种情况下,最简单的方法可能就是让每种类型都成为一个http.Handler自身,并且可以相应地进行调度。

于 2014-12-24T20:32:59.077 回答
0

我最终完全避免了这个问题,我没有尝试减少代码量,而是使用go generateGo 1.4 中的新功能来创建每个资源所需的代码。

于 2015-01-15T15:45:28.437 回答