在包的 `init()` 函数中包含无限 for 循环——明智之举还是不明智之举?
来源:stackoverflow
时间:2024-03-23 21:00:37 404浏览 收藏
在 Go 包的 `init()` 函数中包含无限 for 循环是一个有争议的做法。有些人认为这是明智之举,因为它允许持续的后台处理,例如定期刷新数据或监控事件。然而,其他人则认为这是不明智的,因为它可能会导致长期可维护性问题,例如难以测试和推理。
我想知道在包的 init()
函数中使用无限 for 循环是否是一个坏主意,或者是否应该避免这样做。
是否有人有任何知识或经验,如果可以这样做或应该避免这种情况?
你会在哪里使用它?
这可以用于例如提供来自外部源的一些信息的包,这些信息必须定期刷新(例如每天一次)。
我使用了如下类似的代码,但没有“看门狗”功能。这意味着 init()
刚刚启动了一个 go 例程,该例程将在后台运行,并在 tick 到达时运行更新过程。
不幸的是,这个更新机制在 aprox 之后停止工作。由于未知原因,3 个月了,但服务运行良好,只是使用“旧”数据。
简单示例实现
https://play.golang.org/p/k-gi1t9j4op 上的完整示例
package info import ( "log" "sync" "time" ) var ( data map[string]interface{} lock sync.RWMutex ) func init() { // ticker channel ticker := time.NewTicker(1 * time.Second).C // "watchdog" loop for { log.Println("Starting Update Loop") var wg sync.WaitGroup wg.Add(1) // Start asyc update process. go func() { defer wg.Done() //notify wg when if process ends for whatever reason // Loop forever // Run when a tick is received from the `ticker` channel for { select { case <-ticker: log.Println("Update ticker received") err := update() if err != nil { log.Printf("ERROR: %v\n", err.Error()) } } } }() wg.Wait() } } // internal update function that retrieves some information from some external system func update() error { lock.Lock() defer lock.Unlock() log.Println("Update `data`") // retrieve information and update `data` return nil } // Public function to query data func GetInformation(key string) interface{} { lock.RLock() defer lock.RUnlock() return data[key] }
该代码可以正常工作并在单元测试中运行良好,并且也可以正常运行。 我想知道长期稳定性(一年或更长的正常运行时间)等等。
解决方案
它位于 goroutine 中,因此技术上没有问题,但直接在 init
中实现行为会使其非常难以使用,原因有两个:
- 很难测试,就像
main
很难测试一样。测试init
是否调用另一个函数要容易得多,然后可以对其进行测试。 - 这很难推理。 “自动”发生的事情越多,对于使用该包的任何开发人员(包括未来的你)来说,它的意义就越小。 “好吧,我导入了这个包并使用了一个微小的函数,现在不知怎的,我的 CPU 使用率从 1% 上升到了 50%,我做错了什么”,这样的事情将会出现,需要更多的探索才能弄清楚。
TL;DR 不存在“长期稳定性”问题,但很可能存在长期可维护性问题。
文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《在包的 `init()` 函数中包含无限 for 循环——明智之举还是不明智之举?》文章吧,也可关注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次学习