首页 > 后端开发 > Golang > 逆变可以解决 Go 1.18 的 Pipe 函数中的泛型类型兼容性问题吗?

逆变可以解决 Go 1.18 的 Pipe 函数中的泛型类型兼容性问题吗?

Linda Hamilton
发布: 2024-11-24 07:47:11
原创
381 人浏览过

Can Contravariance Solve Go 1.18's Generic Type Compatibility Issue in the Pipe Function?

Go 1.18 泛型中的逆变类型

在 Go 1.18 中,泛型类型允许定义类型随类型参数而变化的函数。逆变是一种特定类型的方差,它允许函数接受更广泛的输入类型,同时保持输入和输出类型之间定义良好的关系。

管道函数

提供的 Pipe 函数旨在组合转换值的函数。它期望左侧函数的输出类型与右侧函数的输入类型匹配。但是,在下面的示例中,它无法编译:

func Pipe[A, T1, T2 any](left func(A) T1, right func(T1) T2) func(A) T2 {
    return func(a A) T2 {
        return right(left(a))
    }
}

func OpenFile(name string) *os.File {
    ...
}

func ReadAll(rdr io.Reader) []byte {
    ...
}

var OpenRead = Pipe(OpenFile, ReadAll)
登录后复制

编译器会抱怨,因为 T1 被认为是 *os.File,即使它与 io.Reader 兼容。

有解决办法吗?

不幸的是,Go 1.18 没有办法泛型修改 Pipe 的签名以启用所需的行为。 Go 不支持协变结果类型,这意味着函数的输出类型不能根据输入类型而变化。

这是一个 Bug 吗?

中的行为Go 1.18 不是一个 bug。这是设计使然,正如 Go 泛型常见问题解答中所解释的那样。

解决方法

要解决此限制,可以使用 Pipe 函数的修改版本,但是它牺牲了编译时类型安全:

func Pipe[A, T1, T2, T3 any](left func(A) T1, right func(T2) T3) func(A) T3 {
    return func(a A) T3 {
        return right(any(left(a)).(T2))
    }
}
登录后复制

以上是逆变可以解决 Go 1.18 的 Pipe 函数中的泛型类型兼容性问题吗?的详细内容。更多信息请关注PHP中文网其他相关文章!

来源:php.cn
本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
作者最新文章
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板