首页 > 解决方案 > 如何编写惯用的构造函数

问题描述

我对 Go 中的构造函数感到困惑。我见过的大多数构造函数都返回一个结构,但是“Effective Go”表明在某些情况下可以根据“Generality”的规则返回一个接口。

我相信“Effective Go”能提供好的想法,但这似乎没有遵循“接受接口,返回结构”的原则。我猜很多类型都实现了一个接口,仅此而已,所以在这种情况下,通常会看到返回接口的构造函数。

另一个相关的说法是接口应该由消费者定义,但“一般性”意味着接口是由生产者定义的。

有人可以澄清吗?

标签: goconstructor

解决方案


正如已经提到的,返回一个接口应该被认为是特殊的。

返回error接口类型的错误是这些异常之一。

返回表示未导出类型的接口是另一个例外。但是为什么你会有一个描述未导出结构的导出接口,而不是仅仅有一个导出结构呢?

原因很简单,这使您可以更高程度地控制该结构的构造方式。

比较这两段代码:

type MyType struct {
    MyField string
}

func NewMyType(value string) MyType {
    return MyType{value}
}

func (t MyType) MyMethod() string {
    return t.MyField
}
type MyType interface {
    MyMethod() string
}

type myType struct {
    MyField string
}

func NewMyType(value string) MyType {
    return myType{value}
}

func (t myType) MyMethod() string {
    return t.MyField
}

在第一种情况下,我可以这样做:myVar := MyType{}而在第二种情况下,我将无法这样做,我不得不使用提供的构造函数。第一种情况还允许在创建后修改字段值,这在第二种情况下是不允许的。使字段不导出将解决第二部分但不是第一部分。

这个例子显然是微不足道的,但是能够构造无效的结构可能会产生可怕的影响。通过使用特定的构造函数,您可以确保对象处于有效的起始状态,并且您只需要确保它始终处于有效状态。如果您不能确保这一点,您可能需要在每个方法开始时检查它是否处于有效状态。

例如考虑一个数据库请求。它需要一个数据库连接。如果用户能够在没有数据库连接的情况下创建数据库请求,您将必须检查它在每种方法中是否有效。如果您强制他使用构造函数,您可以在创建时检查并完成。


推荐阅读