当前位置: 首页 > article >正文

golang中的错误处理机制

golang中的错误处理机制

在现代编程语言中,错误处理非常重要。因为再高的高手也无法预见程序运行时的状况,想确保程序在遇到问题时能够给出有用的反馈,而不是无声无息地失败。就需要在出错时根据状况判断和处理问题。良好的错误处理不仅有助于提升代码的健壮性,还能增强程序的可维护性和可调试性。golang的错误处理机制与其他主流编程语言有所不同。golang的错误处理机制是一种显式的方式,强调明确的错误检查和处理,而不依赖于异常机制。go 的错误处理设计强调简洁性、可读性和显式控制。

golang并没有采用传统的(如 Java 中的 try-catch 或 python 中的 try-except)机制,而是通过显式的返回值来报告和处理错误。golang也有自己的异常处理机制,采用panic 和 recover,结合defer的语法结构,这个后面再发文分析。今天主要讲述可预见、可恢复的有显式返回值的错误问题。

 1错误类型和基本概念

在 Go 中,错误通常通过实现 error 接口来表示。error 是一个内置的接口类型,定义如下:

type error interface {
	Error() string
}

所有实现了 Error() 方法的类型都可以作为 error 类型。最常见的错误类型是标准库中的 errors.New 和 fmt.Errorf 函数返回的错误对象。

1.1错误的构造

Go 中的错误通常是通过调用 errors.New 或 fmt.Errorf 创建的。

package main

import (
	"errors"

	"fmt"
)

func main() {
	err := errors.New("something went wrong") // 创建一个错误对象

	fmt.Println(err) // 输出:something went wrong

	// 使用 fmt.Errorf 创建带格式化的错误
	err = fmt.Errorf("an error occurred: %s", "details about the error")

	fmt.Println(err) // 输出:an error occurred: details about the error
}

1.2错误的自定义类型

Go 鼓励开发者创建自定义的错误类型,这些类型通常通过结构体实现,结构体实现了 Error() 方法来满足 error 接口。

package main

import "fmt"

// 定义自定义错误类型
type MyError struct {
	Code int

	Message string
}

func (e *MyError) Error() string {
	return fmt.Sprintf("Error %d: %s", e.Code, e.Message)
}

func main() {
	err := &MyError{Code: 404, Message: "Page Not Found"}

	fmt.Println(err) // 输出:Error 404: Page Not Found
}

2错误处理的基本方式

在 Go 中,函数通常返回两个值:一个是结果,另一个是错误。如果函数执行成功,错误值为 nil;如果出现问题,错误值包含了具体的错误信息。调用者需要显式地检查错误值,并采取相应的措施。

2.1错误检查

在 golang中,错误检查通常是通过检查函数返回值的方式进行的。这是 Go 中最基本的错误处理方式。

package main

import (
	"errors"

	"fmt"
)

func divide(a, b int) (int, error) {
	if b == 0 {
		return 0, errors.New("division by zero")
	}

	return a / b, nil
}

func main() {
	result, err := divide(4, 2)

	if err != nil { // 如果发生错误,输出错误信息
		fmt.Println("Error:", err)
	} else {
		fmt.Println("Result:", result) // 没有发生错错误,输出正确的结果
	}

	// 触发错误
	result, err = divide(4, 0)

	if err != nil {
		fmt.Println("Error:", err) // 输出:Error: division by zero
	}
}

2.2多值返回

Go 的函数可以返回多个值,其中之一通常是 error 类型。这要求调用者在接收返回值时,总是检查错误。

package main

import (
	"fmt"

	"errors"
)

func readFile(filename string) (string, error) { // 函数返回值类型为 string 和 error
	if filename == "" {
		return "", errors.New("filename cannot be empty") // 返回错误
	}

	return "file content", nil // 函数返回值
}

func main() {
	content, err := readFile("") // 调用函数,传传入空字符串

	if err != nil { // 判断错误
		fmt.Println("Error:", err) // 输出:Error: filename cannot be empty
	} else {
		fmt.Println("File content:", content) // 输入了正确的参数,才会输出:File content: file content
	}
}

2.3简单的错误处理

golang 提倡显式的错误检查。错误必须被捕捉并适当地处理。对于错误,开发者必须明确是否想要继续执行操作、重试操作、或者返回错误。

package main

import "fmt"

func processData(data string) error { /// 模拟处理数据
	if data == "" { // 如果数据为空,返回错误
		return fmt.Errorf("data cannot be empty")
	}

	// 正常处理数据
	return nil
}

func main() {
	if err := processData(""); err != nil { // 调用处理数据的函数,传入错误的空值
		fmt.Println("Error:", err) // 输出:Error: data cannot be empty
	}
}

3错误处理的最佳实践

3.1尽早失败

在 Go 中,“尽早失败”是一个常见的错误处理原则,即在出现错误时,尽快返回错误,而不是继续执行后续代码。

package main

import "fmt"

func processData(data string) error { // 函数返回错误
	if data == "" { // 验证数据是否为空
		return fmt.Errorf("data cannot be empty")
	}

	// 继续处理数据
	fmt.Println("Processing:", data)
	return nil
}

func main() {
	// 尽早返回错误
	if err := processData(""); err != nil { // 处理错误

		fmt.Println("Error:", err)

		return
	}

	fmt.Println("Success!")
}

3.2封装错误信息

Go 提供了 fmt.Errorf 函数,可以通过格式化字符串封装详细的错误信息。在复杂的系统中,错误信息可能会被封装成不同的层次,以便为调用者提供更多上下文信息。

package main

import "fmt"

func readFile(filename string) error { // 函数返回错误
	return fmt.Errorf("failed to read file %s: %w", filename, fmt.Errorf("file not found")) // 返回一个错误对象
}

func main() {
	err := readFile("config.txt") // 调用函数

	if err != nil { // 判断错误
		fmt.Println("Error:", err) // 输出:Error: failed to read file config.txt: file not found
	}

}

3.3错误包装(Error Wrapping)

Go 1.13 引入了 fmt.Errorf 的 %w 格式说明符,允许错误的“包装”,从而将原始错误附加到新错误上。这样可以保留错误的上下文信息,并且允许后续的错误检查。

package main

import (
	"fmt"

	"errors"
)

func processData(filename string) error { // 函数返回一个错误类型
	err := readFile(filename) // 假设readFile函数返回一个错误

	if err != nil { // 如果readFile返回错误,则处理错误并返回一个新错误
		return fmt.Errorf("processing failed: %w", err)
	}
	return nil
}

func readFile(filename string) error { // 假设readFile函数返回一个错误
	return errors.New("file not found")
}

func main() {
	err := processData("config.txt") // 调用processData函数

	if err != nil { // 如果processData返回错误,则处理错误
		fmt.Println("Error:", err) // 输出:Error: processing failed: file not found
	}
}

通过错误包装,能够追踪到原始错误的来源。

3.4使用错误类型判断错误

golang 提供了 errors.Is 和 errors.As 函数,可以用于检查错误类型和错误是否匹配。这使得错误处理更加灵活,尤其是在复杂系统中。

package main

import (
	"fmt"

	"errors"
)

var ErrNotFound = errors.New("resource not found") // 自定义错误类型
func findResource(name string) error { // 模拟查找资源
	return fmt.Errorf("failed to find resource %s: %w", name, ErrNotFound) // 返回错误
}

func main() {
	err := findResource("example") // 调用查找资源函数

	if errors.Is(err, ErrNotFound) { // 判断是否为自定义错误类型

		fmt.Println("Resource not found error:", err) // 处理自定义错误类型

	} else {

		fmt.Println("Other error:", err)

	}
}

4错误处理的总结

显式错误处理:Go 中的错误处理方式要求开发者显式地检查返回的错误值。这种方式简单、清晰,鼓励开发者主动处理可能出现的问题。

错误返回值:Go 函数通过返回值来报告错误。错误值通常为 nil(表示成功)或包含具体的错误信息。

错误包装:Go 提供了 fmt.Errorf 和 errors.Is、errors.As 等工具来处理错误包装和错误类型判断,增强了错误处理的灵活性。

尽早失败:在面对错误时,尽早退出函数,避免继续执行不必要的操作。

错误类型:可以通过自定义错误类型来表示更复杂的错误,使得错误信息更加明确和具有上下文。


http://www.kler.cn/a/461397.html

相关文章:

  • 【数据结构】树链刨分
  • 弧形导轨如何避免生锈?
  • SpringCloud源码分析-nacos与eureka
  • VSCode设置ctrl或alt+mouse(left)跳转
  • 《Xsens动捕与人形机器人训练》讲座将于1月9日下午2:30在线上召开
  • SpringBoot异步线程@Async的使用注意
  • Fetch处理大模型流式数据请求与解析
  • OpenLinkSaas使用手册-项目外部资源管理
  • HarmonyOS:@Require装饰器:校验构造传参
  • 深入解析 Android MediaHTTPConnection JNI 实现
  • 2024广东省职业技能大赛云计算——私有云(OpenStack)平台搭建
  • Java Web学生自习管理系统
  • 课程设计项目之基于Python实现围棋游戏代码
  • REDIS1.0
  • 【每日学点鸿蒙知识】长时任务、HarmonyAppProvision申请、preferences、Testing工具、应用保活
  • 2.ATK-DLRK3568 QT竖屏显示改为横屏显示
  • 【MySQL初级】第1-4章
  • quasar中@click.stop没有生效,点击按钮时候会跳转
  • 【2024年-9月-29日-开源社区openEuler实践记录】 Euler - Copilot - Framework:开启智能辅助编程新征程
  • Rabbitmq追问1
  • Go语言中值接收者和指针接收者的区别?
  • HTML<select>标签有关的定义和属性
  • 【人工智能机器学习基础篇】——深入详解监督学习之模型评估:掌握评估指标(准确率、精确率、召回率、F1分数等)和交叉验证技术
  • c# Record关键字
  • Github 正常访问但是ping不同也无法进行git操作
  • 通过无障碍服务(AccessibilityService)实现Android设备全局水印显示