我有以下Go代码。我想将Foo
结构或Bar
类型作为字符串处理。使用“句柄”,我的意思是我想将/它/它的底层值转换为(真实)类型string
。我有一个解决方法,但我觉得它在结构的情况下不直观。
寻找类型(而不是结构)似乎是更好的方法。我不需要维护任何状态,也不需要“继承”类型特定的功能,所以在我的情况下它应该工作。但是,对type.String()
的调用会导致堆栈递归。如果我没有遗漏某些东西(很明显),我很好奇。
package main
import "fmt"
type Foo struct {
string
}
func (f *Foo) String() string {
return f.string
}
type Bar string
func (b *Bar) String() string {
return fmt.Sprintf("%s", b) // Cannot use: return string(b) here.
}
func main() {
a := Foo{"a"}
var b Bar
b = "b"
fmt.Printf("A is: %s\n", a) // Doesn't call a.String() ?
//fmt.Printf("A is: %s\n", string(a)) // Doesn't work
fmt.Printf("A is: %s\n", a.string) // workaround A
fmt.Printf("A is: %s\n", a.String()) // workaround B, required if I want to use it in a different package
fmt.Printf("B is: %s\n", b) // Calls b.String()
fmt.Printf("B is: %s\n", string(b))
//fmt.Printf("B is: %s\n", b.String()) // Causes a stack overflow
}
输出:
A is: {a}
A is: a
A is: a
B is: b
B is: b
Go的游乐场代码:https://play.golang.org/p/zgrKao4cxa 行为来自Go 1.5.2版
以下是基于原始问题答案的简短示例。此外,以下帖子有助于理解和推理该主题:Value receiver vs. Pointer receiver in Golang?
如果是类型,则以下作品:
type MyString string
func (b MyString) String() string {
return string(b)
}
Go的Playground链接:https://play.golang.org/p/H12bteAk8D
如果是结构,以下工作:
package main
import "fmt"
type MyString struct {
string
someState int
}
func (m MyString) String() string {
return string(m.string)
}
func main() {
// The verbose version:
//var a MyString = MyString{string: "a", someState: 1}
a := MyString{"a", 1}
fmt.Printf("A is: %s\n", a)
fmt.Printf("A is: %s\n", a.String())
}
Go的Playground链接:https://play.golang.org/p/GEKeY4rmB8
答案 0 :(得分:0)
你已经为String
方法创建了一个指针接收器,但是你正在使用值,而不是指向它们的指针,所以它不适用。您需要切换到指针或更改String
方法签名:
package main
import "fmt"
type Foo struct {
string
}
func (f Foo) String() string {
return "My " + f.string
}
type Bar string
func (b Bar) String() string {
return fmt.Sprintf("My %s", string(b))
}
func main() {
a := Foo{"a"}
var b Bar = "b"
fmt.Printf("A is: %s\n", a)
fmt.Printf("B is: %s\n", b)
}
请小心receivers type:
关于接收器的指针与值的规则是值方法 可以在指针和值上调用,但指针方法只能是 在指针上调用
哦,还有一件事。如果已定义,fmt.Sprintf("%s", b)
会调用String
方法。所以,你会得到一个递归。