Is it possible to multiplex several channels into one?

后端 未结 3 1864
北海茫月
北海茫月 2021-02-14 00:29

The idea is to have a variable number of channels in a slice, push each value received through them into a single channel, and close this output channel once the last one of the

3条回答
  •  借酒劲吻你
    2021-02-14 01:03

    Edit: added pairwise reduction example code and reordered parts of the answer.

    The preferred solution is the non-answer of "restructure so that you don't have a slice of channels." The restructuring can often make use of the feature that multiple goroutines can send to a single channel. So instead of having each of your sources send on separate channels and then having to deal with receiving from a bunch of channels, just create one channel and let all the sources send on that channel.

    Go does not offer a feature to receive from a slice of channels. It's a frequently asked question, and while the solution just given is preferred, there are ways to program it. A solution I thought you were suggesting in your original question by saying "reducing the slice pairwise" is a binary divide and conquer solution. This works just fine, as long as you have a solution for multiplexing two channels into one. Your example code for this is very close to working.

    You're just missing one little trick to make your example code work. Where you decrement n, add a line to set the channel variable to nil. For example, I made the code read

        case v, ok := <-cin1:
            if ok {
                cout <- v
            } else {
                n--
                cin1 = nil
            }
        case v, ok := <-cin2:
            if ok {
                cout <- v
            } else {
                n--
                cin2 = nil
            }
        }
    

    This solution does what you want and is not busy waiting.

    So then, a full example incorporating this solution into a function that multiplexes a slice:

    package main
    
    import (
        "fmt"
        "time"
    )
    
    func multiplex(cin []chan int, cout chan int) {
        var cin0, cin1 chan int
        switch len(cin) {
        case 2:
            cin1 = cin[1]
            fallthrough
        case 1:
            cin0 = cin[0]
        case 0:
        default:
            cin0 = make(chan int)
            cin1 = make(chan int)
            half := len(cin) / 2
            go multiplex(cin[:half], cin0)
            go multiplex(cin[half:], cin1)
        }
        for cin0 != nil || cin1 != nil {
            select {
            case v, ok := <-cin0:
                if ok {
                    cout <- v
                } else {
                    cin0 = nil
                }
            case v, ok := <-cin1:
                if ok {
                    cout <- v
                } else {
                    cin1 = nil
                }
            }
        }
        close(cout)
    }
    
    func main() {
        cin := []chan int{
            make(chan int),
            make(chan int),
            make(chan int),
        }
        cout := make(chan int)
        for i, c := range cin {
            go func(x int, cx chan int) {
                for i := 1; i <= 3; i++ {
                    time.Sleep(100 * time.Millisecond)
                    cx <- x*10 + i
                }
                close(cx)
            }(i, c)
        }
        go multiplex(cin, cout)
        for {
            select {
            case v, ok := <-cout:
                if ok {
                    fmt.Println("main gets", v)
                } else {
                    return
                }
            }
        }
    }
    

提交回复
热议问题