2

我想为我的框架制作模型,用 go 编写,但我不确定如何以共享常见数据库交互方法的方式组合它们:保存、更新、删除。

我通常会通过为所有具体模型创建一个模型抽象父类来做到这一点,但 Go 没有继承。您应该改用嵌入和组合,但我不知道如何嵌入模型类并让它保存持有它的类的数据。

我看到了另一种选择,即创建一个在其中嵌入具体模型类型的模型类,但我并没有真正看到适用于所有模型的接口,除非它是空的。这带来了任何东西都可以被视为模型的不安全感。

做什么?

4

1 回答 1

4

在我的项目中,我会这样做:

type Storable interface {
    // called after unmarshalling from the database
    Init() error
    // called when an object is being deleted
    // this is useful if the object needs to delete other objects,
    // change state on a remote server, etc.
    Destroy() error
    // called after Init, helps separate initialization from
    // sanity checks (useful to detect errors before using a potentially
    // invalid object)
    Validate() error
    // type of this object, stored in the database in `Save` and `Update`
    // so it can be read out in `Get`
    Type() string
}

如果您使用的是 SQL 数据库,则可以执行以下操作:

type Schema map[string]reflect.Type

type SQLStorable interface {
    Storable
    Schema() Schema
}

然后在数据库中,我有这样的功能:

func Get(id string) (Storable, error)
func Save(Storable) error
func Update(id string, Storable) error
func Delete(id string) error
// register a type with the database (corresponds to the Type() in Storable)
func Register(typ string, reflect.Type)

我在数据库中保留了一个对象缓存:map[string]Storable. 这允许我实现缓存逻辑以减少查找时间(不需要每次从数据库中读取对象时都重新构建对象)。

在我的项目中,我有很多包需要与其他包中的对象进行通信。由于管理依赖链将是一场噩梦,因此我建立了一个使用数据库的消息传递系统:

type Message map[string]interface{}
func Send(id string, Message)

我已经Receive向 Storable 添加了一个函数,它接受 aMessage并返回一个错误。到目前为止,这减少了许多令人头疼的问题,并带来了更可插拔的设计。

我不确定这是否是“Go way”,但它避免了继承的想法并解决了问题。在数据库逻辑中,我使用大量反射从数据库中获取数据并用它填充对象。它会导致一些不幸的类型断言,但我想在试图保持抽象时真的无济于事。

于 2013-07-21T02:08:59.180 回答