Go开发中的常见错误有哪些
这篇文章主要介绍"Go开发中的常见错误有哪些",在日常操作中,相信很多人在Go开发中的常见错误有哪些问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答"Go开发中的常见错误有哪些"的疑惑有所帮助!接下来,请跟着小编一起来学习吧!
1.无限递归调用
递归调用自身的函数需要有一个退出条件。否则,它将永远递归,直到系统内存耗尽。
此问题可能是由常见错误引起的,例如忘记添加退出条件。它也可能"故意"发生。某些语言具有尾调用优化,这使得某些无限递归调用可以安全使用。尾调用优化允许您避免为函数分配新的堆栈帧,因为调用函数将返回它从被调用函数获取的值。最常见的用途是尾递归,其中为利用尾调用优化而编写的递归函数可以使用常量堆栈空间。然而,Go 并没有实现尾调用优化,你最终会耗尽内存。然而,这个问题不适用于产生新的 goroutine。
2. 分配给nil地图
在添加任何元素之前,需要使用make函数(或map文字)初始化映射。使用内置函数创建一个新的空映射值make,该函数将map类型和可选的容量提示作为参数:
make(map[string]int)make(map[string]int, 100)
初始容量不限制其大小:地图增长以容纳存储在其中的项目数量,nil
地图除外。甲nil
地图相当于不同之处在于可以添加没有元素的空映射。
不好的模式:
var countedData map[string][]ChartElement
好的模式:
countedData := make(map[string][]ChartElement)
3.方法修改接收器
修改非指针接收器值的方法可能会产生不良后果。这是一个错误风险,因为该方法可能会更改方法内部接收器的值,但不会反映在原始值中。要传播更改,接收者必须是一个指针。
例如:
type data struct { num int key *string items map[string]bool}func (d data) vmethod() { d.num = 8}func (d data) run() { d.vmethod() fmt.Printf("%+v", d) // Output: {num:1 key:0xc0000961e0 items:map[1:true]}}
如果num
必须修改:
type data struct { num int key *string items map[string]bool}func (d *data) vmethod() { d.num = 8}func (d *data) run() { d.vmethod() fmt.Printf("%+v", d) // Output: &{num:8 key:0xc00010a040 items:map[1:true]}}
4. Goroutine 中可能使用了不需要的值
循环中的范围变量在每次迭代中都被重用;因此,在循环中创建的 goroutine 将指向上作用域的范围变量。这样,goroutine 就可以使用带有不需要的值的变量。
在下面的示例中,goroutine 中使用的 index 和 value 的值来自外部范围。因为 goroutine 是异步运行的,所以 index 和 value 的值可能(通常是)与预期值不同。
mySlice := []string{"A", "B", "C"}for index, value := range mySlice { go func() { fmt.Printf("Index: %d\n", index) fmt.Printf("Value: %s\n", value) }()}
为了克服这个问题,必须创建一个本地作用域,如下例所示。
mySlice := []string{"A", "B", "C"}for index, value := range mySlice { index := index value := value go func() { fmt.Printf("Index: %d\n", index) fmt.Printf("Value: %s\n", value) }()}
处理此问题的另一种方法是将值作为 args 传递给 goroutine。
mySlice := []string{"A", "B", "C"}for index, value := range mySlice { go func(index int, value string) { fmt.Printf("Index: %d\n", index) fmt.Printf("Value: %s\n", value) }(index, value)}
5.Close在检查可能的错误之前推迟
对于实现接口的值defer的Close()方法,这是 Go 开发人员的常见模式io.Closer。例如,打开文件时:
f, err := os.Open("/tmp/file.md")if err != nil { return err}defer f.Close()
但是这种模式对于可写文件是有害的,因为推迟函数调用会忽略其返回值,并且该Close()方法可能会返回错误。例如,如果您将数据写入文件,则在您调用Close. 应明确处理此错误。
虽然您可以在不使用的情况下继续,但defer您需要记住每次完成工作时关闭文件。更好的方法是defer使用包装函数,如下例所示。
f, err := os.Open("/tmp/file.md")if err != nil { return err}defer func() { closeErr := f.Close() if closeErr != nil { if err == nil { err = closeErr } else { log.Println("Error occured while closing the file :", closeErr) } }}()return err
在团队中工作时,审查其他人的代码变得很重要。DeepSource是一种自动化代码审查工具,可管理端到端代码扫描过程,并在推送新提交或新拉取请求时自动发出带有修复的拉取请求。
为 Go 设置 DeepSource非常简单。一旦设置完成,将对整个代码库执行初始扫描,找到改进的范围,修复它们,并为这些更改打开 PR。
到此,关于"Go开发中的常见错误有哪些"的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注网站,小编会继续努力为大家带来更多实用的文章!