如何将用户定义的命名类型/结构转换为匿名类型

时间:2016-01-25 10:34:18

标签: go

原始问题

我有以下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

1 个答案:

答案 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方法。所以,你会得到一个递归。