Go 中嵌入类型的 String() 方法的不明确行为
在处理 Go 中的嵌入类型时,了解 String 的行为() 方法可能会令人费解。让我们根据特定的代码示例深入研究此行为。
考虑以下代码:
type Engineer struct { Person TaxPayer Specialization string } type Person struct { Name string Age int } func (p Person) String() string { return fmt.Sprintf("name: %s, age: %d", p.Name, p.Age) } type TaxPayer struct { TaxBracket int } func (t TaxPayer) String() string { return fmt.Sprintf("%d", t.TaxBracket) } func main() { engineer := Engineer{ Person: Person{ Name: "John Doe", Age: 35, }, TaxPayer: TaxPayer{3}, Specialization: "Construction", } fmt.Println(engineer) }
此代码的输出是:
{name: John Doe, age: 35 3 Construction}
但是,如果删除 Person.String(),输出将变为:
3
并且如果TaxPayer.String() 也被删除,输出更改为:
{{John Doe 35} {3} Construction}
最初,似乎 Engineer 结构必须有一个隐式 String() 方法。然而,事实并非如此。
嵌入类型中的 String() 方法
当类型嵌入到结构体中时,可以通过以下方式访问它们的字段和方法:嵌入类型。如果多个嵌入类型定义同名的方法(例如 String()),这种方法的“提升”可能会导致歧义。
在给定的代码示例中,因为 Person 和 TaxPayer 都有一个 String()方法,将这些方法提升到 Engineer 类型会导致歧义。这就是为什么 Engineer.String() 会导致编译错误。
为什么使用 fmt.Println() 时不会出现歧义错误
尽管 Engineer 的方法集中有歧义、 fmt.Println(engineer) 不会导致编译错误。这是因为 fmt.Println() 调用 fmt.Fprint(os.Stdout, Engineer)。
fmt.Fprint() 检查传递的值是否实现 fmt.Stringer 接口,其中包含 String() 方法。如果是,则使用 String() 生成值的字符串表示形式。
在这种情况下,由于 Person 和 TaxPayer 都没有实现 fmt.Stringer,因此使用默认格式(结构字段)。这会导致我们在调用 fmt.Println(engineer) 时看到的输出。
结论
理解嵌入类型的行为及其方法的推广至关重要在围棋中。当多个嵌入类型定义同名的方法时,可能会导致歧义,从而导致编译错误。但是,使用 fmt.Println() 时,如果传递的值未实现 fmt.Stringer,则使用默认格式。
以上是当嵌入类型具有冲突的'String()”方法时,为什么'fmt.Println()”与 Go 中的嵌入类型的行为不同?的详细内容。更多信息请关注PHP中文网其他相关文章!