Here is a version using RWLock https://play.golang.org/p/YOwuYFiqtlf <https://play.golang.org/p/YOwuYFiqtlf>
It won’t run correctly in the playground because it terminates when all routines are asleep - which happens during the test (not sure why it does this, as sleeping is different than a deadlock). It is probably less efficient, and less orderly than the other example using WaitGroup but you get the idea I hope. It forcibly terminates the writers before they complete by design. > On Aug 28, 2019, at 4:09 PM, Michel Levieux <m.levi...@capitaldata.fr> wrote: > > One should also be careful regarding the conceptual demands he or she is > making. > Having a shared resource (that is complex enough that it cannot be atomically > accessed or modified) means essentially that "having multiple writers being > transparent to the readers", fundamentally, is not possible. > > From the moment itself when such a resource is shared, there must be some > sort of mecanism (that one using resources atomically usable) that ensures > the integrity of it. > Maybe what you're talking about is having it transparent in terms of code, in > which case we both agree, but if you're looking for something transparent in > essence, as in performance, logical construction and all the rest, I think > there is a misunderstanding here: even if it was added in the language, there > would be many many things going on under the hood, as it is already (and > cannot really be otherwise) for channel use alone. > > As for the priority using selects, I think it's more of something to be dealt > with on the "user-side". There are many kinds of priority in general, and > trying to implement something in the language itself would IMO either be too > specific compared to the nessecary time to do so or it would probably have a > huge overhead on the "classical' use case of the select construct. > + the fact that it is apparently already possible using RWMutexes. > > Le mer. 28 août 2019 à 22:37, Marcin Romaszewicz <marc...@gmail.com > <mailto:marc...@gmail.com>> a écrit : > Think of a channel as existing for the lifetime of a particular data stream, > and not have it be associated with either producer or consumer. Here's an > example: > > https://play.golang.org/p/aEAXXtz2X1g <https://play.golang.org/p/aEAXXtz2X1g> > > The channel here is closed after all producers have exited, and all consumers > continue to run until the channel is drained of data. > > The producers are managed by something somewhere in your code - and that is > the scope at which it makes sense to create channel ownership. I've used a > waitgroup to ensure that the channel is closed after all producers exit, but > you can use whatever barrier construct you want. > > Even if you must have a channel per producer, you can safely close the > producer side, without notifying the downstream about this. The example early > in the thread uses multiple channels, with one channel being used to signal > that the producers should exit. Channels aren't really the right model for > this, you want a thread safe flag of some sort. For example: > > var exitFlag uint64 > func producer(chan data int, wg *sync.WaitGroup) { > defer wg.Done() > for { > shouldExit := atomic.LoadUint64(&exitFlag) > if shouldExit == 1 { > return > } > chan <- rand.Intn(100) > } > } > > Here's 10 producers and 3 consumers sharing a channel and closing it safely > upon receiving an exit flag: > https://play.golang.org/p/RiKi1PGVSvF <https://play.golang.org/p/RiKi1PGVSvF> > > -- Marcin > > On Wed, Aug 28, 2019 at 11:29 AM Leo Lara <l...@leopoldolara.com > <mailto:l...@leopoldolara.com>> wrote: > I do not think priority select is *necessary*, it could be a nice addition if > the performance does not change. > > On Wednesday, August 28, 2019 at 8:27:36 PM UTC+2, Leo Lara wrote: > Hi Robert, > > From the article: """To bound more the problem, in my case, you control the > writers but not the readers""" > > So what I was trying to do was to be able to close, with mutiple writers, > while being transparent for the readers. The readers only need to read as > usual form the channel. > > For example, if you want to write a library where the user just reads from a > channel, this is an approach I found where the user of the lirbary deos nto > have to do anything special. Of course, there might be another solution, but > if you need to modify the reader we are talking about a different problem. > > Cheers!! > > On Wednesday, August 28, 2019 at 7:17:24 PM UTC+2, Robert Engels wrote: > A better solution is to wrap the writes using a RWLock, grab the read lock > for writing, and the Write lock for closing. Pretty simple. > > Just encapsulate it all in a MultiWriterChannel struct - generics would help > here :) > > -----Original Message----- > From: Leo Lara > Sent: Aug 28, 2019 11:24 AM > To: golang-nuts > Subject: [go-nuts] Re: An old problem: lack of priority select cases > > This is connected with my article: > https://dev.to/leolara/closing-a-go-channel-written-by-several-goroutines-52j2 > > <https://dev.to/leolara/closing-a-go-channel-written-by-several-goroutines-52j2> > > I think there I show it is possible to workaround that limitation using > standard Go tools. Of course, the code would be simple with priority select, > but also perhaps select would become less efficient. > > On Wednesday, August 28, 2019 at 6:06:33 PM UTC+2, T L wrote: > The old thread: > https://groups.google.com/forum/#!topic/golang-nuts/ZrVIhHCrR9o > <https://groups.google.com/forum/#!topic/golang-nuts/ZrVIhHCrR9o> > > Go channels are flexible, but in practice, I often encountered some > situations in which channel are hard to use. > Given an example: > > import "math/rand" > > type Producer struct { > data chan int > closed chan struct{} > } > > func NewProducer() *Producer { > p := &Producer { > data: make(chan int), > closed: make(chan struct{}), > } > > go p.run() > > return p > } > > func (p *Produce) Stream() chan int { > return p.data > } > > func (p *Producer) run() { > for { > // If non-blocking cases are selected by their appearance order, > // then the following slect block is a perfect use. > select { > case(0) <-p.closed: return > case p.data <- rand.Int(): > } > } > } > > func (p *Produce) Clsoe() { > close(p.closed) > close(p.data) > } > > func main() { > p := NewProducer() > for n := p.Stream() { > // use n ... > } > } > > > If the first case in the select block in the above example has a higher > priority than the second one, > then coding will be much happier for the use cases like the above one. > > In short, the above use case requires: > * for receivers, data streaming end is notified by the close of a channel. > * for senders, data will never be sent to closed channel. > > But, as Go 1 doesn't support priority select cases, it is much tedious to > implement the code > satisfying the above listed requirements. The final implementation is often > very ugly and inefficient. > > Does anyone else also experience the pain? > > -- > You received this message because you are subscribed to the Google Groups > "golang-nuts" group. > To unsubscribe from this group and stop receiving emails from it, send an > email to golan...@googlegroups.com <>. > To view this discussion on the web visit > https://groups.google.com/d/msgid/golang-nuts/b284f880-034a-4721-8686-ef48d3e2c14c%40googlegroups.com > > <https://groups.google.com/d/msgid/golang-nuts/b284f880-034a-4721-8686-ef48d3e2c14c%40googlegroups.com?utm_medium=email&utm_source=footer>. > > > > > -- > You received this message because you are subscribed to the Google Groups > "golang-nuts" group. > To unsubscribe from this group and stop receiving emails from it, send an > email to golang-nuts+unsubscr...@googlegroups.com > <mailto:golang-nuts+unsubscr...@googlegroups.com>. > To view this discussion on the web visit > https://groups.google.com/d/msgid/golang-nuts/aeb38a0a-8268-42d7-a8eb-ce5ef01c5380%40googlegroups.com > > <https://groups.google.com/d/msgid/golang-nuts/aeb38a0a-8268-42d7-a8eb-ce5ef01c5380%40googlegroups.com?utm_medium=email&utm_source=footer>. > > -- > You received this message because you are subscribed to the Google Groups > "golang-nuts" group. > To unsubscribe from this group and stop receiving emails from it, send an > email to golang-nuts+unsubscr...@googlegroups.com > <mailto:golang-nuts+unsubscr...@googlegroups.com>. > To view this discussion on the web visit > https://groups.google.com/d/msgid/golang-nuts/CA%2Bv29LvcUhUvrZb_8AKYWj0A%2Bqd5LKBPmbz-RVBb%3DJn_gNZE6w%40mail.gmail.com > > <https://groups.google.com/d/msgid/golang-nuts/CA%2Bv29LvcUhUvrZb_8AKYWj0A%2Bqd5LKBPmbz-RVBb%3DJn_gNZE6w%40mail.gmail.com?utm_medium=email&utm_source=footer>. > > -- > You received this message because you are subscribed to the Google Groups > "golang-nuts" group. > To unsubscribe from this group and stop receiving emails from it, send an > email to golang-nuts+unsubscr...@googlegroups.com > <mailto:golang-nuts+unsubscr...@googlegroups.com>. > To view this discussion on the web visit > https://groups.google.com/d/msgid/golang-nuts/CANgi337s1Low95QvqJUAOTsqcVji7uMQ_jr%3DFftpt2uMz5_XSQ%40mail.gmail.com > > <https://groups.google.com/d/msgid/golang-nuts/CANgi337s1Low95QvqJUAOTsqcVji7uMQ_jr%3DFftpt2uMz5_XSQ%40mail.gmail.com?utm_medium=email&utm_source=footer>. -- You received this message because you are subscribed to the Google Groups "golang-nuts" group. To unsubscribe from this group and stop receiving emails from it, send an email to golang-nuts+unsubscr...@googlegroups.com. To view this discussion on the web visit https://groups.google.com/d/msgid/golang-nuts/54CB637F-BA52-4834-B081-E5C9FF82C284%40ix.netcom.com.