go语言异常处理

error接口

go语言引入了一个关于错误错里的标准模式,即error接口,该接口的定义如下:

type error interface{
Error() string
}

对于要返回错误的大多数函数来说,大致上都可以定义为如下的模式,将error作为多个返回值中的最后一个,但是这并非是强制要求

func Foo(param int) (n int,err error){
//
}

调用该函数的时候建议按照如下的方式处理错误情况

n,err := Foo(0)
if err != nil{
//处理错误
}else{
//处理结果
} 那么,如何自定义error类型呢,我们以Go库中的实际代码为例来说明,首先,定义一个承载错误信息的类型,因为Go中的接口非常的灵活,你根本不需要像别的语言一样使用继承或则implements来明确指定类型和接口之间的关系,代码如下:

type PathError struct{

Op string

Path string

Err error

}

这样定义后,编译器如何知道PathError可以当成一个error来传递呢?关键在于下面的代码实现了Error() 方法:

func (e *PathError) Error() string{
return e.Op + " " + e.Path +" " + e.Err.Error()
} 定义了上述方法之后,就可以直接返回PathError变量了,如下:

func test(name string) (fi FIleInfo,err error){

var stat syscall.Stat_t

err = syscall.Stat(name,&stat)

if err != nil{

return nil,$PathError{"stat",name,err}

}

return file,nil

}

如果在处理错误的时候要获取错误的详细信息,就需要用到类型的转换知识了:

fi,err := os.Stat("a.txt")
if err != nil{
if e,ok := err.(*os.PathError);ok && e.Err != nil{
//获取PathError类型变量e总的其他信息
}
}

defer

关键字defer是Go引入的一个很有意思的特性,他能保证在函数中发生异常的情况下,defer定义的语句仍然会被执行,如下:

func CopyFile(dst,src string) (w int64,err error){
srcFile,err := os.Open(src)
if err != nil{
return
}
defer srcFile.Close()
dstFile,err := os.Create(dst)
if err != nil{
return
}
defer dstFile.Close()
return io.Copy(dstFile,srcFile)
}

即使Copy函数抛出异常,Go仍然会保证文件被正常的关闭,如果据的一句话干不完清理工作的话,可以在defer后加一个匿名函数:

defer func(){
//
}()

值得注意的是一个函数中可以定义多个defer,并且defer语句的顺序是按照先进后出的顺序执行的,也就是说最后一个defer语句将最先被执行

panic()和recover()

GO引入了两个内置的函数panic()和recover()以报告和处理运行时错误和程序中的错误场景

func panic(interface{})
func recover() interface{}

当在一个函数中调用panic函数后,正确的执行流程会被立即终止,但函数中之前使用defer关键字延迟执行的语句将正常展开执行,之后该函数将返回到调用函数,并导致逐步向上执行panic流程,直到所属的goroutine中所有正在执行的函数被终止,错误信息将被报告,这个过程成为错误处理流程

从panic中传入的interface{}中我们可以看出,该函数接收任意类型的数据

recover用于终止错误处理流程,一般情况下,recover应该在一个使用defer关键字的函数中执行以有效截取错误处理流程,如果在发生异常的goroutine中没有调用recover,会导致该goroutine所属的进程直接退出

下面是一个常用的场景

defer func(){
if r := recover();r!=nil{
log.Printf("Runtime error caught: %v",r)
}
}

go语言异常处理的更多相关文章

  1. 第63课 C语言异常处理

    1. 异常的概念 (1)程序在运行过程中可能产生异常 (2)异常(Exception)与Bug的区别 ①异常是程序运行时可预料的执行分支 ②Bug是程序是的错误,是不被预期的运行方式 2. 异常和Bu ...

  2. Go语言异常处理defer\panic\recover

    Go语言追求简洁优雅,所以,Go语言不支持传统的 try…catch…finally 这种异常,因为Go语言的设计者们认为,将异常与控制结构混在一起会很容易使得代码变得混乱.因为开发者很容易滥用异常, ...

  3. c语言异常处理机制

    异常处理机制:setjmp()函数与longjmp()函数 C标准库提供两个特殊的函数:setjmp() 及 longjmp(),这两个函数是结构化异常的基础,正是利用这两个函数的特性来实现异常. 所 ...

  4. C语言异常处理和连接数据库

    #include <stdio.h> #include <setjmp.h> jmp_buf j; void Exception(void); double diva(doub ...

  5. C语言异常处理之 setjmp()和longjmp()

    异常处理之除0情况 相信大家处理除0时,都会通过函数,然后判断除数是否为0,代码如下所示: double divide(doublea,double b) { const double delta = ...

  6. C语言异常处理编程的三个境界

    http://blog.csdn.net/treefish2012/article/details/17466487 这是上一次看完Herb Sutter的<Exceptional C++> ...

  7. C语言异常处理

    异常的概念-程序在运行过程中可能产生异常-异常(Exception)与Bug的区别 异常是程序运行时可预料的执行分支 Bug是程序中的错误,是不被预期的运行方式 异常(Exception)和Bug的对 ...

  8. GO语言异常处理03---自定义异常

    package main import ( "fmt" "time" ) /* type error interface { Error() string } ...

  9. GO语言异常处理02---返回错误

    package main import ( "errors" "fmt" "math" ) /*计算圆面积,同时返回[结果-错误对]*/ f ...

随机推荐

  1. Python3 timeit的用法

    Python3中的timeit模块可以用来测试小段代码的运行时间 其中主要通过两个函数来实现:timeit和repeat,代码如下: def timeit(stmt="pass", ...

  2. CSS: Grid homework redact.

    The web homework: Finished design: (I use six block with different color to show this homework and I ...

  3. 解决no module named ipykernel_launcher

    解决no module named ipykernel_launcher 最近开hydrogen的时候,提示no module named ipykernel_launcher. 记得以前解决过这个问 ...

  4. electron-vue 淘宝源

    yarn-js config set registry https://registry.npm.taobao.org

  5. 微软BI 之SSIS 系列 - 数据仓库中实现 Slowly Changing Dimension 缓慢渐变维度的三种方式

    开篇介绍 关于 Slowly Changing Dimension 缓慢渐变维度的理论概念请参看 数据仓库系列 - 缓慢渐变维度 (Slowly Changing Dimension) 常见的三种类型 ...

  6. 【iOS】Objective-C 字符串操作

    字符串中查找指定字符方法 //开头 -(BOOL) hasPrefix:(NSString *)aString; //结尾 -(BOOL) hasSuffix:(NSString *)aString; ...

  7. elk中es集群web管理工具cerebro

    cerebo是kopf在es5上的替代者 安装es虽然不能再root下运行,但是cerebro 可以 run as root is ok wget https://github.com/lmeneze ...

  8. 通过启动函数定位main()函数

      如下,通过vc6.0编写一个hello world程序.尝试结合汇编代码.分析启动函数找到main函数.   在printf(xxx)插入断点,调试执行.如下,在堆栈窗口中可见main()下的一个 ...

  9. [leetcode]25. Reverse Nodes in k-Group每k个节点反转一下

    Given a linked list, reverse the nodes of a linked list k at a time and return its modified list. k  ...

  10. SpringMVC 使用验证框架 Bean Validation(上)

    SpringMVC 使用验证框架 Bean Validation(上) 对于任何一个应用而言在客户端做的数据有效性验证都不是安全有效的,这时候就要求我们在开发的时候在服务端也对数据的有效性进行验证. ...