在不使用switch语句的情况下在运行时进行选择实现

时间:2019-02-10 11:21:15

标签: go

我想使用提供的字符串在运行时选择接口的实现。我不想使用switch语句-代码应该是通用的,并且可以与实现接口的任何新结构一起使用,而无需修改(打开/关闭)。

假设我具有以下结构:

type Fooer interface {
    Foo()
}

type A struct{}

func (_ *A) Foo() {
    fmt.Println("Calling A")
}

type B struct{}

func (_ *B) Foo() {
    fmt.Println("Calling B")
}

type C struct{}

func (_ *C) Foo() {
    fmt.Println("Calling C")
}

然后,我想做类似的事情:

func main() {
    // let's pretend it's user input
    input := []string{"C", "A", "C"}

    for _, className := range input {
        // struct := StructFromName(className)
        // struct.Foo()
    }
}

要打印:

Calling C
Calling A
Calling C

https://play.golang.org/p/mOW5miz5LdU

令我失望的是,运行时go不保留结构名称的注册表,因此没有StructFromName(className)可用。我也不想自己创建这样的注册表。 免责声明:我主要用Java编写并且有OOP习惯,所以我希望简单到

Class<?> clazz = Class.forName(className);
Object object = clazz.newInstance();
Foo foo = (Foo) object;
foo.Foo();

我尝试了另一种方法,使用方法而不是接口,它可以工作,但是感觉很笨拙且不习惯:

type Hack struct{}

func (_ *Hack) FooA() {
    fmt.Println("Calling A")
}

func (_ *Hack) FooB() {
    fmt.Println("Calling B")
}

func (_ *Hack) FooC() {
    fmt.Println("Calling C")
}

func main() {
    hack := &Hack{}

    // let's pretend it's user input
    input := []string{"FooC", "FooA", "FooC"}

    for _, funcName := range input {
        reflect.ValueOf(hack).MethodByName(funcName).Call(nil)
    }
}

https://play.golang.org/p/3ueEuVV2Hx9

有更好的方法吗?

1 个答案:

答案 0 :(得分:2)

解决这类问题的最佳方法确实是注册表。我建议使用标准库的database/sql包中的注册表作为指导。

每个SQL驱动程序(MySQL,PostgreSQL等)都通过调用sql.Register在其自己的init()函数中注册。

这将创建干净的依赖关系树(无循环依赖关系),并允许您的类型的临时实现,甚至可能来自第三方。