延迟执行函数
来源:stackoverflow
时间:2024-03-01 11:51:22 218浏览 收藏
有志者,事竟成!如果你在学习Golang,那么本文《延迟执行函数》,就很适合你!文章讲解的知识点主要包括,若是你对本文感兴趣,或者是想搞懂其中某个知识点,就请你继续往下看吧~
我正在学习golang源代码并陷入延迟函数执行顺序。 我有两个文件:一个定义端点的行为,另一个用于测试。我删除了一些与我的问题无关的代码,以减少需要阅读的行数。 端点定义文件
// endpoint is the fundamental building block of servers and clients. // it represents a single rpc method. type endpoint func(ctx context.context, request interface{}) (response interface{}, err error) // middleware is a chainable behavior modifier for endpoints. type middleware func(endpoint) endpoint // chain is a helper function for composing middlewares. requests will // traverse them in the order they're declared. that is, the first middleware // is treated as the outermost middleware. func chain(outer middleware, others ...middleware) middleware { return func(next endpoint) endpoint { for i := len(others) - 1; i >= 0; i-- { // reverse next = others[i](next) } return outer(next) } }
测试文件包含打印的步骤。
func examplechain() { e := endpoint.chain( annotate("first"), annotate("second"), annotate("third"), )(myendpoint) if _, err := e(ctx, req); err != nil { panic(err) } // output: // first pre // second pre // third pre // my endpoint! // third post // second post // first post } var ( ctx = context.background() req = struct{}{} ) func annotate(s string) endpoint.middleware { return func(next endpoint.endpoint) endpoint.endpoint { return func(ctx context.context, request interface{}) (interface{}, error) { fmt.println(s, "pre") defer fmt.println(s, "post") return next(ctx, request) } } } func myendpoint(context.context, interface{}) (interface{}, error) { fmt.println("my endpoint!") return struct{}{}, nil }
据我了解,应该首先执行 annotate
三个方法,然后执行 endpoint.chain
方法,最后执行 myendpoint
方法。另外,由于首先打印 pre
,并且当函数返回“post”时,应根据 go 文档中的 defer
解释进行遵循:
“defer”语句调用一个函数,该函数的执行被推迟到周围函数返回的那一刻,要么是因为周围函数执行了 return 语句,到达了其函数体的末尾,要么是因为相应的 goroutine 出现了恐慌。
所以我期望看到的是
// Output: // first pre // first post // second pre // second post // third pre // third post // my endpoint!
简而言之,我的问题是:
- 为什么
first pre
后面没有first post
,与second
third
相同。 post
s 的顺序相反。endpoint.chain
反转annotate
返回值列表的执行,但首先评估annotate
方法,对吗?更不用说,打印了pre
s,这意味着首先执行内部函数
正确答案
延迟函数作为函数中的最后一件事运行,位于 return 语句之后,因此 annotate
函数将首先运行 next
,只有在返回后延迟函数才会运行。根据您的代码,它应该打印的顺序是:
first pre second pre third pre my endpoint third post second post first post
Here is your example turned into something that runs on the Go playground.
请注意,如果您在给定函数中调用 defer
多次,则每个延迟调用均按 lifo 顺序运行。因此,如果您想使用 defer
确保您的 post
首先被调用,然后 next
运行,请考虑替换:
defer fmt.println(s, "post") next(ctx, request)
与:
defer next(ctx, request) defer fmt.println(s, "post)
当然,在您的情况下,您希望返回 next
返回的内容,这会产生一个小问题。要在实际情况下解决这个问题,您需要一个小函数和一些命名的返回值:
defer func() { i, e = next(ctx, request) }()
其中 i
和 e
是指定的返回值。
Here is that same code turned into a new example in which the deferred calls occur in the desired order. 在这种情况下,这个例子相当愚蠢,因为没有任何恐慌,并且中间没有“危险步骤”,所以我们真正需要的是顺序执行两个 fmt.println
调用,而不使用 defer
。但是,如果我们可以在 fmt.println(s, "pre")
和 post 部分之间恐慌,那么这可能是有意义的。
好了,本文到此结束,带大家了解了《延迟执行函数》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多Golang知识!
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
477 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习