- xml - AJAX/Jquery XML 解析
- 具有多重继承的 XML 模式
- .net - 枚举序列化 Json 与 XML
- XML 简单类型、简单内容、复杂类型、复杂内容
一直在看《用go构建微服务》,书中介绍了 apache/go-resiliency/deadline
用于处理超时的包。
deadline.go
// Package deadline implements the deadline (also known as "timeout") resiliency pattern for Go.
package deadline
import (
"errors"
"time"
)
// ErrTimedOut is the error returned from Run when the deadline expires.
var ErrTimedOut = errors.New("timed out waiting for function to finish")
// Deadline implements the deadline/timeout resiliency pattern.
type Deadline struct {
timeout time.Duration
}
// New constructs a new Deadline with the given timeout.
func New(timeout time.Duration) *Deadline {
return &Deadline{
timeout: timeout,
}
}
// Run runs the given function, passing it a stopper channel. If the deadline passes before
// the function finishes executing, Run returns ErrTimeOut to the caller and closes the stopper
// channel so that the work function can attempt to exit gracefully. It does not (and cannot)
// simply kill the running function, so if it doesn't respect the stopper channel then it may
// keep running after the deadline passes. If the function finishes before the deadline, then
// the return value of the function is returned from Run.
func (d *Deadline) Run(work func(<-chan struct{}) error) error {
result := make(chan error)
stopper := make(chan struct{})
go func() {
result <- work(stopper)
}()
select {
case ret := <-result:
return ret
case <-time.After(d.timeout):
close(stopper)
return ErrTimedOut
}
}
deadline_test.go
package deadline
import (
"errors"
"testing"
"time"
)
func takesFiveMillis(stopper <-chan struct{}) error {
time.Sleep(5 * time.Millisecond)
return nil
}
func takesTwentyMillis(stopper <-chan struct{}) error {
time.Sleep(20 * time.Millisecond)
return nil
}
func returnsError(stopper <-chan struct{}) error {
return errors.New("foo")
}
func TestDeadline(t *testing.T) {
dl := New(10 * time.Millisecond)
if err := dl.Run(takesFiveMillis); err != nil {
t.Error(err)
}
if err := dl.Run(takesTwentyMillis); err != ErrTimedOut {
t.Error(err)
}
if err := dl.Run(returnsError); err.Error() != "foo" {
t.Error(err)
}
done := make(chan struct{})
err := dl.Run(func(stopper <-chan struct{}) error {
<-stopper
close(done)
return nil
})
if err != ErrTimedOut {
t.Error(err)
}
<-done
}
func ExampleDeadline() {
dl := New(1 * time.Second)
err := dl.Run(func(stopper <-chan struct{}) error {
// do something possibly slow
// check stopper function and give up if timed out
return nil
})
switch err {
case ErrTimedOut:
// execution took too long, oops
default:
// some other error
}
}
// in deadline_test.go
if err := dl.Run(takesTwentyMillis); err != ErrTimedOut {
t.Error(err)
}
我无法理解上述代码的执行流程。据我了解,因为 takesTwentyMillis
函数休眠时间超过设置的超时持续时间 10 毫秒,
// in deadline.go
case <-time.After(d.timeout):
close(stopper)
return ErrTimedOut
time.After 发出当前时间,并选择这种情况。然后关闭 stopper channel 并返回 ErrTimeout。
我不明白的是,关闭停止 channel 对可能仍在运行的匿名 goroutine 有何影响我认为,当停止器 channel 关闭时,下面的 goroutine 可能仍在运行。
go func() {
result <- work(stopper)
}()
(这里说错了请指正)我想在close(stopper)
之后,这个 goroutine 会调用 takesTwentyMillis
(=功函数)以塞子 channel 作为其参数。该函数将继续并休眠 20 毫秒并返回 nil 以传递给结果 channel 。 main() 到此结束,对吧?
我不明白在这里关闭塞子 channel 有什么意义。 takesTwentyMillis
函数似乎并没有在函数体内使用 channel :(。
// in deadline_test.go within TestDeadline()
done := make(chan struct{})
err := dl.Run(func(stopper <-chan struct{}) error {
<-stopper
close(done)
return nil
})
if err != ErrTimedOut {
t.Error(err)
}
<-done
这是我不完全理解的部分。我想什么时候dl.Run
运行时,停止 channel 被初始化。但是因为stopper channel中没有值,函数调用会阻塞在<-stopper
处。 ...但是因为我不理解这段代码,所以我不明白为什么这段代码首先存在(即这段代码试图测试什么,以及它是如何执行的,等等)。
所以我明白当Run
第二个问题中的函数触发停止器 channel 关闭,工作函数获取信号。然后 worker 关闭 done channel 并返回 nil。我使用 delve(=go debugger) 来查看,gdb 将我带到 deadline.go
中的 goroutine。在 return nil
行之后.
err := dl.Run(func(stopper <-chan struct{}) error {
<-stopper
close(done)
--> return nil
})
在输入 n 跳到下一行后,delve 将我带到这里
go func() {
--> result <- work(stopper)
}()
过程在这里结束了,因为当我再次输入 n 时,命令行提示 PASS 并且过程退出。为什么这个过程在这里结束? work(stopper)
似乎返回 nil
,然后应该将其传递给结果 channel ,对吗?但是由于某种原因,这一行似乎没有执行。
我知道主 goroutine,它是 Run
函数,已经返回 ErrTimedOut。所以我想这与此有关?
最佳答案
第一个问题
stopper
的使用 channel 用于发出功能信号,例如takesTwentyMillis
它的截止日期已经到了,调用者不再关心它的结果。通常这意味着 worker 函数像 takesTwentyMillis
应该检查 stopper
channel 已经关闭,因此它可能会取消它的工作。仍然,检查 stopper
channel 是辅助功能的选择。它可能会或可能不会检查 channel 。
func takesTwentyMillis(stopper <-chan struct{}) error {
for i := 0; i < 20; i++ {
select {
case <-stopper:
// caller doesn't care anymore might as well stop working
return nil
case <-time.After(time.Second): // simulating work
}
}
// work is done
return nil
}
第二个问题
这部分Deadline.Run()
将关闭塞子 channel 。
case <-time.After(d.timeout):
close(stopper)
在关闭的 channel ( <-stopper
) 上读取将立即返回该 channel 的零值。我认为它只是在测试最终超时的工作函数。
关于go - 当 channel 关闭时,以接收 channel 作为参数的 goroutines 是否停止?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53673690/
发件人:http://blog.nindalf.com/how-goroutines-work/ As the goroutines are scheduled cooperatively, a go
很多时候在用 Go 开发 http 服务器时,我都会遇到这种困境。 假设我想尽快用http statuscode 200响应客户端(然后在后面执行工作),这就是我通常这样做的原因: 我让我的主要 ht
这是代码: import "fmt" func main() { messages := make(chan string, 1) go func(c chan string) {
我正在学习 Golang,但遇到了一些困难。我已经研究过 Google,但没有任何进展。 我编写了一个代码,通过多台服务器的 ICMP 检查 RTT。 它有这样的结构: type Server str
我想运行多个 goroutine,进行一些处理,将结果放入 channel ,当至少有一个 goroutine 完成时,完成所有其他 goroutine 并从 channel 返回结果。 所以,我尝试
我有两个(但以后我会是三个)go 例程来处理来自远程服务器(来自 ampq channel )的传入消息。但是因为它们正在处理相同的数据/状态,所以我想阻止所有其他 go 例程,除了正在运行的例程。
我有一个案例,我从 2 个不同的位置(ES 和 REDIS)读取数据,我需要从最快的源读取一个值,因此我触发了 2 个 goroutines,一个从 ES 获取数据,其他从REDIS获取。 一旦从其中
像这里一样,我创建了一个 go playground 示例:sGgxEh40ev ,但无法正常工作。 quit := make(chan bool) res := make(chan int) go
我是golang的新手,正在研究goroutine。 我写了一个简单的代码,故意使用 goroutine 来划分数字。 首先,我给出基数并继续除它的数,直到它不能被整除 但是,我改变了go split
Main { go routine_1(carryout a time consuming task and return output) go routine_2(wait for output f
我想知道从另一个 goroutine 返回时调用的 goroutine 会发生什么。他们是继续运行还是被终止?这是一个示例代码来说明我的意思: func func() { // Doing s
更具体地说,在我的例子中,我有一个网络服务器和一个全局可访问的结构,网络服务器使用它来生成页面。我有另一个 Goroutine,它总是定期用新值更新该结构。这会引起问题吗?我是否需要实现一种机制来确保
来自 this file ,我不明白为什么函数startWorker会这样写: func (p *WorkerPool) dispatch() { for i := 0; i < p.maxW
我正在学习围棋,但在使用 goroutines 时遇到了问题。这是我的代码 package main import ( "fmt" "sync" "time" ) var co
我收到以下错误,我不明白为什么: 发送:查询 Herefatal 错误:所有 goroutines 都睡着了 - 死锁! 您可以看到我正在调用我使用 goroutine 创建的函数 routine。我
大家好,我正在从 Python3 过渡到 Go,所以我正在尝试重写我创建的库以获得更好的性能。 我面临一个问题,因为我是 Golang XD 中的新手,我使用有限的 API 下载数百个 json,我想
我有以下格式的脚本部分: func main() { for i=0;i<1000000;i++ { go test() } } func test() { a := test
package main func main() { c:=make(chan int) for i:=0; i<=100;i++ {
我正在学习 Go,我的第一个项目是一个简单的 ping 脚本。本质上,我想 ping 一堆 url,并在每个响应时等待 XXX 秒,然后再次 ping。这是删减的代码: func mai
这个问题在这里已经有了答案: Go all goroutines are asleep deadlock (2 个回答) fatal error: all goroutines are asleep
我是一名优秀的程序员,十分优秀!