- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我已尝试实现 go 服务器的正常关闭,如博文 http://grisha.org/blog/2014/06/03/graceful-restart-in-golang/ 中所述.主要内容如下。
自定义监听器:
var httpWg sync.WaitGroup // initialised in the other part
type gracefulListener struct {
net.Listener
stop chan error
stopped bool
}
func newGracefulListener(l net.Listener) (gl *gracefulListener) {
gl = &gracefulListener{Listener: l, stop: make(chan error)}
go func() {
_ = <-gl.stop
gl.stopped = true
gl.stop <- gl.Listener.Close()
}()
return
}
func (gl *gracefulListener) Accept() (c net.Conn, err error) {
c, err = gl.Listener.Accept()
if err != nil {
return
}
c = gracefulConn{Conn: c} // wrap using our custom connection
httpWg.Add(1) // increase the counter
return
}
func (gl *gracefulListener) Close() error {
if gl.stopped {
return syscall.EINVAL
}
gl.stop <- nil
return <-gl.stop
}
func (gl *gracefulListener) File() *os.File {
tl := gl.Listener.(*net.TCPListener)
fl, _ := tl.File()
return fl
}
自定义连接:
type gracefulConn struct {
net.Conn
}
func (w gracefulConn) Close() error {
httpWg.Done() // <- panics sometimes
return w.Conn.Close()
}
想法是当程序收到 SIGTERM 时,它会停止为新连接提供服务,并等待现有连接的 httpWg.Wait()
完成。这种方法在本地有效,但当我部署它时,有时我会在 httpWg.Done()
行的 gracefulConn.Close()
中收到 panic :
panic: sync: negative WaitGroup counter
panic 不是在我停止服务器时发生的,而是在例行服务期间发生的。怎么可能有更多的 Close()
调用然后 Accept()
调用?还是我遗漏了什么?
附言我尝试将 stopped
属性和互斥量添加到 gracefullConn
,因此在 Close
中它会锁定互斥量并检查 stopped
以确保我们只停止一次。但是,我仍然收到同样的 panic 。
最佳答案
Close()
可以被多次调用,因此您肯定需要在 func (w gracefulConn) Close() error
中检查它。
P.S. I have tried to add stopped property and a mutex to gracefullConn, so in Close it locks the mutex and checks stopped to ensure we stop it only once. However, I still received the same panic.
请记住,如果 gracefulConn
作为值而不是引用传递,那么任何互斥锁/标志都不会按预期工作。所以一定要将c = gracefulConn{Conn: c}
变成c = &gracefulConn{Conn: c}
。
关于去优雅关闭负WaitGroup,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33413747/
我想了解 Goroutine 中的同步。我这里有一个代码,它在一个 channel 上写入从 0 到 4 的数字,完成后我使用 range 从 channel 读取并打印值。 下面的代码在我等待时使用
我正在尝试使用递归函数查找所有目录的列表。该函数的代码是 func FindDirs(dir string, nativePartitions []int64, wg *sync.WaitGroup,
1. 简介 本文将介绍 Go 语言中的 WaitGroup 并发原语,包括 WaitGroup 的基本使用方法、实现原理、使用注意事项以及常见的使用方式。能够更好地理解和应用 WaitGrou
这个问题在这里已经有了答案: golang - Content of a pointer in a slice changes during recursive function run (1 个回答
我做了一个简单的代码示例来理解pipeline的用法,就在这里。 package main import ( "fmt" "sync" "time" ) func main()
谁能给我一个简单的示例代码来告诉我如何Wait Group有效或可以实现(如果给出例如 this 会更好)?谢谢。 最佳答案 我认为您的示例无法实现 WaitGroup (或同步互斥锁)。 关于go
我用 Go 编写了一些代码来解析站点并检索所有链接及其 Http 响应。我的代码运行良好,但我想添加 GoRoutines 以查看它在递归函数中的工作方式。 package main import (
我编写了以下代码,以便运行直到有人手动退出程序。 确实是 -----每1秒检查一次是否存在 -----如果可用则读取文件并逐行打印文件内容 为此,我首先从 main 调用了一个函数然后我调用一个 Wa
我遇到了一些并发问题。我是第一次编写并发应用程序。 我正在努力实现的目标 依赖函数(使用 goroutines)即 func2 依赖于 func1 问题 如果我在完成waiting后重用waitgro
在下面的代码中,如果其中一个启动的 go 例程花费太长时间(例如 > 10 秒)完成,我如何添加适当的超时错误处理?请注意,我不想有一个“整体”超时,而是每个 go 例程的超时,这样我也可以在我的错误
我已经在stdout和stderr goroutine中添加了以下内容(Go func()…),以等待获取output并完成error。 现在我希望外部函数将等待,直到两个例程都完成为止。 func
我想了解我围绕 WaitGroups 的逻辑是否正确,并看看是否有更有效的方式来构建我的代码。目的是尽可能快地执行任务。 我的代码填充了 _urls通过标准输入填充的 channel 。然后我启动了两
在我的 Go 程序中,我为每个部门启动了多个工作组。 我想在退出程序之前等待每个部门的工作人员完成 我不能使用单个 WaitGroups,因为在实际场景中,我可能必须结束任何特定部门并且只需要等待。
func check(name string) string { resp, err := http.Get(endpoint + name) if err != nil {
我有这个简单的脚本,试图遍历文件系统并逐行读取文件以匹配正则表达式上的行: package main import ( "bufio" "fmt" "io/ioutil" "log"
我之前在goroutine中使用sync.WaitGroup,但是我想控制goroutine并发性, 所以我用并发限制写我的等待组: package wglimit import ( "syn
可能是我看不到明显的东西,我做错了什么: func printSize (listOfUrls []string){ var wg sync.WaitGroup wg.Add(len(
可能是我看不到明显的东西,我做错了什么: func printSize (listOfUrls []string){ var wg sync.WaitGroup wg.Add(len(
我正在尝试并行运行递归快速排序调用: func quicksort(a []int) { quicksortRecursive(a) wg.Wait() insertionso
我有一个依赖于并发检查某些错误的函数,我正在尝试使用 WaitGroup 等待所有返回可能错误的进程完成,然后再检查所有错误。 它似乎跳过了一些 wg.Done() cals。这是调试的 youtub
我是一名优秀的程序员,十分优秀!