如何在没有竞争的情况下延长计时器的持续时间?

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

How to extend the ticker duration without a race?

问题

我正在尝试实现一个保持连接的机制。问题在于,我不知道如何在没有竞争的情况下替换保持连接的计时器(conn.keepAlive),因为keepAlive()方法总是从计时器中读取。

//为了简洁起见,未处理错误
const interval = 10 * time.Second

type conn struct {
keepAlive time.Ticker
conn net.Conn
mux sync.Mutex
}

// replace替换底层连接
func (cn conn) replace(newcn net.Conn) {
cn.mux.Lock()
cn.conn = newcn
// 重置计时器
cn.keepAlive.Stop
cn.keepAlive = time.NewTicker(interval)
cn.mux.Unlock()
}

func (cn conn) keepAlive() {
for {
<-cn.keepAlive.C
cn.mux.Lock()
cn.conn.Write([]byte("ping"))
var msg []byte
cn.conn.Read(msg)
if string(msg) != "pong" {
// 做一些坏事
}
cn.keepAlive = time.NewTicker(interval)
cn.mux.Unlock()
}
}

英文:

I'm trying to implement a keepAlive mechanism. The issue is that I don't know how to replace the keep alive ticker ( conn.keepAlive) without a race because keepAlive() method always reads from the ticker.

  1. //errors not handled for brevity
  2. const interval = 10 * time.Second
  3. type conn struct {
  4. keepAlive time.Ticker
  5. conn net.Conn
  6. mux sync.Mutex
  7. }
  8. // replace replaces the underlying connection
  9. func (cn conn) replace(newcn net.Conn) {
  10. cn.mux.Lock()
  11. cn.conn = newcn
  12. // reset the ticker
  13. cn.keepAlive.Stop
  14. cn.keepAlive = time.NewTicker(interval)
  15. cn.mux.Unlock()
  16. }
  17. func (cn conn) keepAlive() {
  18. for {
  19. &lt;-cn.keepAlive.C
  20. cn.mux.Lock()
  21. cn.conn.Write([]byte(&quot;ping&quot;))
  22. var msg []byte
  23. cn.conn.Read(msg)
  24. if string(msg) != &quot;pong&quot; {
  25. // do some mean stuff
  26. }
  27. cn.keepAlive = time.NewTicker(interval)
  28. cn.mux.Unlock()
  29. }
  30. }

答案1

得分: 0

一种更简洁的实现方式是使用通道作为同步机制,而不是互斥锁:

  1. type conn struct {
  2. sync.Mutex
  3. conn net.Conn
  4. replaceConn chan net.Conn
  5. }
  6. // replace replaces the underlying connection
  7. func (cn *conn) replace(newcn net.Conn) {
  8. cn.replaceConn <- newcn
  9. }
  10. func (cn *conn) keepAlive() {
  11. t := time.NewTicker(interval)
  12. msg := make([]byte, 10)
  13. for {
  14. select {
  15. case <-t.C:
  16. case newConn := <-cn.replaceConn:
  17. cn.Lock()
  18. cn.conn = newConn
  19. cn.Unlock()
  20. continue
  21. }
  22. cn.Lock()
  23. _ = msg
  24. // do keepalive
  25. cn.Unlock()
  26. }
  27. }

以上是代码的翻译结果。

英文:

One way to implement this much more succinctly, is by using a channel as the synchronization mechanism, instead of the mutex:

  1. type conn struct {
  2. sync.Mutex
  3. conn net.Conn
  4. replaceConn chan net.Conn
  5. }
  6. // replace replaces the underlying connection
  7. func (cn *conn) replace(newcn net.Conn) {
  8. cn.replaceConn &lt;- newcn
  9. }
  10. func (cn *conn) keepAlive() {
  11. t := time.NewTicker(interval)
  12. msg := make([]byte, 10)
  13. for {
  14. select {
  15. case &lt;-t.C:
  16. case newConn := &lt;-cn.replaceConn:
  17. cn.Lock()
  18. cn.conn = newConn
  19. cn.Unlock()
  20. continue
  21. }
  22. cn.Lock()
  23. _ = msg
  24. // do keepalive
  25. cn.Unlock()
  26. }
  27. }

答案2

得分: -2

我最终得到了下面的代码。我对它的外观不太满意,但它是有效的。基本上,我在一个通道中包装了mux,这样我就可以在其上进行选择。

  1. const interval = 10 * time.Second
  2. type conn struct {
  3. keepAlive time.Ticker
  4. conn *net.Conn
  5. mux sync.Mutex
  6. }
  7. // replace replaces the underlying connection
  8. func (cn conn) replace(newcn *net.Conn) {
  9. cn.mux.Lock()
  10. cn.conn = newcn
  11. // reset the ticker
  12. cn.keepAlive.Stop()
  13. cn.keepAlive = time.NewTicker(interval)
  14. cn.mux.Unlock()
  15. }
  16. func (cn conn) keepAlive() {
  17. lockerFn := func() <-chan struct{} {
  18. cn.mux.Lock()
  19. ch := make(chan struct{})
  20. go func() {
  21. ch <- struct{}{}
  22. }()
  23. return ch
  24. }
  25. for {
  26. locker := lockerFn()
  27. select {
  28. case <-cn.keepAlive.C:
  29. // unlock the locker otherwise we
  30. // get stuck
  31. go func() {
  32. <-locker
  33. cn.mux.Unlock()
  34. }()
  35. case <-locker:
  36. cn.conn.Write([]byte("ping"))
  37. var msg []byte
  38. cn.conn.Read(msg)
  39. cn.keepAlive = time.NewTicker(interval)
  40. cn.mux.Unlock()
  41. }
  42. }
  43. }

以上是翻译好的代码部分。

英文:

I ended up with the code below. I'm not very happy how it looks like but it works. Basically I wrapped the mux in a channel so that I can do a select on it.

  1. const interval = 10 * time.Second
  2. type conn struct {
  3. keepAlive time.Ticker
  4. conn *net.Conn
  5. mux sync.Mutex
  6. }
  7. // replace replaces the underlying connection
  8. func (cn conn) replace(newcn *net.Conn) {
  9. cn.mux.Lock()
  10. cn.conn = newcn
  11. // reset the ticker
  12. cn.keepAlive.Stop
  13. cn.keepAlive = time.NewTicker(interval)
  14. cn.mux.Unlock()
  15. }
  16. func (cn conn) keepAlive() {
  17. lockerFn := func() &lt;-chan struct{} {
  18. cn.mux.Lock()
  19. ch = make(chan struct{})
  20. go func() {
  21. ch &lt;- struct{}{}
  22. }()
  23. return ch
  24. }
  25. for {
  26. locker := lockerFn()
  27. select {
  28. case &lt;-cn.keepAlive.C:
  29. // unlock the locker otherwise we
  30. // get stuck
  31. go func() {
  32. &lt;-locker
  33. cn.mux.Unlock()
  34. }()
  35. case &lt;-locker:
  36. cn.conn.Write([]byte(&quot;ping&quot;))
  37. var msg []byte
  38. cn.conn.Read(msg)
  39. cn.keepAlive = time.NewTicker(interval)
  40. cn.mux.Unlock()
  41. }
  42. }
  43. }

huangapple
  • 本文由 发表于 2015年8月13日 03:00:02
  • 转载请务必保留本文链接:https://go.coder-hub.com/31973345.html
匿名

发表评论

匿名网友

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

确定