How to define a type constraint for any channel type

huangapple go评论127阅读模式
英文:

How to define a type constraint for any channel type

问题

我正在尝试定义一个函数,该函数返回给定通道使用率的百分比:

  1. func ChannelUsagePct[T any](channel chan T) int {
  2. channelCap := cap(channel)
  3. channelLen := len(channel)
  4. if channelCap == 0 {
  5. // 无缓冲通道
  6. return 0
  7. }
  8. return 100 * channelLen / channelCap
  9. }
  10. func main() {
  11. twoWayChannel := make(chan int, 10)
  12. var sendOnlyChannel chan<- int = twoWayChannel
  13. var recvOnlyChannel <-chan int = twoWayChannel
  14. fmt.Println(ChannelUsagePct(twoWayChannel))
  15. fmt.Println(ChannelUsagePct(sendOnlyChannel)) // 不起作用
  16. fmt.Println(ChannelUsagePct(recvOnlyChannel)) // 不起作用
  17. }

现在的问题是,这个函数对于第一个通道有效,但对于第二个和第三个通道无效,而不改变函数的签名。但是,如果我改变签名,我需要选择通道是只接收还是只发送。有没有一种定义零通道的方法?例如,不能读取也不能写入,我只想在其上使用caplen函数。
理想情况下,ChannelUsagePct应该适用于这三种情况中的任何一种。

英文:

I'm trying to define a funcion that returns the percentage of usage of a given channel:

  1. func ChannelUsagePct[T any](channel chan T) int {
  2. channelCap := cap(channel)
  3. channelLen := len(channel)
  4. if channelCap == 0 {
  5. // Unbuffered channel
  6. return 0
  7. }
  8. return 100 * channelLen / channelCap
  9. }
  10. func main() {
  11. twoWayChannel := make(chan int, 10)
  12. var sendOnlyChannel chan&lt;- int = twoWayChannel
  13. var recvOnlyChannel &lt;-chan int = twoWayChannel
  14. fmt.Println(ChannelUsagePct(twoWayChannel))
  15. fmt.Println(ChannelUsagePct(sendOnlyChannel)) // Does not work
  16. fmt.Println(ChannelUsagePct(recvOnlyChannel)) // Does not work
  17. }

Now the thing is that this works for the first channel but not for the second and third, without changing the func's signature. But if I change the signature, I need to either choose the channel to be receive only, or send only. Is there a way to define a channel that is zero-way? E.g. cannot be read nor written to, I just want to use cap and len funcs on it.
Ideally the ChannelUsagePct should work for any of the 3 cases.

答案1

得分: 3

为了允许通道的三种可能形式,需要定义一个单独的类型约束。

  1. type Chan[T any] interface {
  2. chan T | <-chan T | chan<- T
  3. }

当使用时,需要使用类型参数将类型约束实例化为T的类型:

  1. func Cap[T any, C Chan[T]](c C) int {
  2. return cap(c)
  3. }

由于实例化的存在,无法推断整个类型,只需要提供通道元素类型:

  1. ch := make(chan int, 3)
  2. i := Cap[int](ch)
  3. fmt.Println(i)
  4. // 3

https://go.dev/play/p/1lx5R4MFvFx

英文:

In order to allow all three possible forms of a channel, a separate type constraint needs to be defined.

  1. type Chan[T any] interface {
  2. chan T | &lt;-chan T | chan&lt;- T
  3. }

This will require instantiating the type constraint with the type of T when it's used, which can be done with another type parameter:

  1. func Cap[T any, C Chan[T]](c C) int {
  2. return cap(c)
  3. }

Because of that instantiation the entire type cannot be inferred, but only the channel element type needs to be supplied:

  1. ch := make(chan int, 3)
  2. i := Cap[int](ch)
  3. fmt.Println(i)
  4. // 3

https://go.dev/play/p/1lx5R4MFvFx

huangapple
  • 本文由 发表于 2023年7月1日 00:31:25
  • 转载请务必保留本文链接:https://go.coder-hub.com/76590753.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定