我知道Go不是面向对象的语言,但我正在尝试按照this文章的建议在我的Iris控制器中实现继承结构。我这样做的主要动机是避免重复。到目前为止,它一直在为我工作。例如,请查看以下代码。
// APIController.go (package Controllers)
type APIController struct {
mvc.C
}
func (c *APIController) Post(data map[string][]string) ([]byte, error) {
data_parsed := c.ParseFormData(data)
return json.Marshal(data_parsed)
}
// UserController.go (package Controllers)
type UserController struct {
mvc.C
*APIController
}
func (c *UserController) Post() ([]byte, error) {
return c.APIController.Post(c.Ctx.FormValues())
}
到目前为止一切顺利。
但我发现很难为模型复制相同的策略。这就是我到目前为止所做的事情
// Model.go (package Models)
type Model struct {
Id string `json:"_id"`
Created_at string `json:"created_at"`
Updated_at string `json:"updated_at"`
Deleted_at string `json:"deleted_at"`
}
// implements further set of functions to be used by 'child' models...
// User.go (package Models)
type User struct {
*Model
First_name string `json:"first_name"`
Last_name string `json:"last_name"`
Email string `json:"email"`
Username string `json:"username"`
Password string `json:"password"`
Last_login string `json:"last_login"`
}
// APIController.go (package Controllers)
type APIController struct {
mvc.C
Model Models.Model
}
// UserController.go (package Controllers)
type UserController struct {
mvc.C
*APIController
}
func (c *UserController) Post() ([]byte, error) {
c.APIController.Model = new(Models.User) //WRONG!
return c.APIController.Post(c.Ctx.FormValues())
}
如您所见,当UserController传递Models.Model
时,APIController期望类型为*Models.User
。最终目标是在APIController中有一个通用模型,来自任何控制器的任何模型然后能够调用Models.Model中定义的所有函数,这样我就不必在任何地方调用这些函数。
有可能这样做吗?如果不是,那么避免重复代码的最佳方法是什么?
更新
通过使用继承和单亲模型并在父APIController
中使用它,我想避免复制我的过滤器/ CRUD逻辑。例如,在UserController
内,如果我想要保存记录,那么User.Save(input_data)
应该在UserController
内定义,而不是在Save
内使用Models.Model
。从APIController
开始,我可以调用Model.Save(input_data)
,而不是单独从子控制器拨打同一个电话。
答案 0 :(得分:1)
首先使Model
成为接口而不是结构。它包含所有模型应该具有的所有方法:
type Model interface {
// Common methods
}
然后,只要User
实现所有这些方法,就可以拥有
c.APIController.Model = new(Models.User) // Now works!
其中一种常见方法可能是Save
。否则,使Save
成为非方法:
func Save(m Model) error {
...
}