gpt4 book ai didi

go - 使用接口(interface)定义的回调参数,使用实现调用

转载 作者:IT王子 更新时间:2023-10-29 01:38:37 26 4
gpt4 key购买 nike

我正在尝试做的事情:

我有一个库包,它定义了一些类型,所有类型都实现了给定的接口(interface)。在整个代码中,都涉及到回调,我没有在所有地方都写回调类型,而是为它定义了一个类型。

type Foo interface {
Bar()
}

type MyCallback func(f Foo)

type CoolFoo int

type BadFoo int

func (cf *CoolFoo) Bar(cb MyCallback) {
}

func (bf *BadFoo) Bar(cb MyCallback) {
}

然后,稍后从使用该库的客户端代码中,我想使用回调进行调用。如果我使用它工作的接口(interface)类型调用它:

cf := &CoolFoo{}

cf.Bar(func(f packageName.Foo) {
})

但我宁愿在我的 IDE 中有更多的 self 记录代码和适当的类型提示,所以我尝试使用实现者类型来调用它,例如:

cf := &CoolFoo{}

cf.Bar(func(f packageName.CoolFoo) {
})

编译失败,报错:

cannot use func literal (type func(packageName.CoolFoo)) as type packageName.MyCallback in argument to cf.Bar

这是不可能的,还是我犯了一些愚蠢的错误?我在 go dev 方面不是很有经验,我试过环顾四周,但在这里找不到解决方案。

我发现将它作为 Foo 或接口(interface){}传递,然后将回调转换为我想要的,但我想避免它,因为它感觉很乱

感谢您的帮助

最佳答案

函数签名是 func(f Foo),因此正是必须传递的内容。原因很简单:如果您的方法需要该签名的回调函数,它可以传入任何实现 Foo 的类型。如果您传入一个只接受 CoolFoo 的函数,调用者仍然可以传入 BadFoo,因为您定义的类型允许这样做。因此,编译器要求类型精确匹配。同样,如果您的回调函数试图转换为具体类型,但那不是传递给它的类型,它将失败,因为您正在做一个代码不支持的假设。

如果您的回调将只传递 CoolFoo,也许这应该是您在签名中使用的类型。如果 CoolFooBadFoo 将各自只将各自的类型传递给它们的回调,也许您需要两个不同的回调签名。如果它们真的可以互换,那么您描述的情况(具体采用 CoolFoo 的函数)不是语言问题,而是您的设计问题。

关于go - 使用接口(interface)定义的回调参数,使用实现调用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44808946/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com