Go语言进阶之路(四):标准错误和异常

标准错误

Go语言内置的error接口,自定义的类型,只要实现该接口方法即可称为标准错误类型,来看看源码:
// The error built-in interface type is the conventional interface for
// representing an error condition, with the nil value representing no error.
type error interface {
  Error() string
}

自定义一个错误类型,实现error接口的Error()方法:

type MyError struct {
  s string
}
func (myError MyError) Error() string {
  return "MyError"
}
func main() {
  var e MyError = MyError{"err"}
  fmt.Println(e)  // 输出:MyError
  err, ok := interface{}(e).(error)
  fmt.Println(ok)  // 输出:true
  fmt.Println(err)  // 输出:MyError
}

从上面例子来看,err, ok := interface{}(e).(error)中的类型转换结果为true,说明自定义的类型就是error的子类型,也就证实了只要某个类型实现了Error()方法,那这个类型就是error类型。

errors包

Go语言的errors包有个内置的错误类型叫errorString,来看一下源码:
package errors


// New returns an error that formats as the given text.
// Each call to New returns a distinct error value even if the text is identical.
func New(text string) error {
  return &errorString{text}
}


// errorString is a trivial implementation of error.
type errorString struct {
  s string
}


func (e *errorString) Error() string {
  return e.s
}

从上面的源码看到,errorString类型实现了error接口的Error()方法,因此errorString类型就是error的子类型。


我们知道,当某个包下面的变量和方法名以小写开头时,这个变量或者方法只能在这个包下面才能访问。因此,我们自己写的代码没办法直接创建errorString的实例,我们可以使用errors里的New方法方便的创建error类型。

var e = errors.New("MyError")
fmt.Println(e)  // 输出:MyError
err, ok := interface{}(e).(error)
fmt.Println(ok)  // 输出:true
fmt.Println(err)  // 输出:MyError

和最上面的例子一样,err, ok := interface{}(e).(error)中的类型转换结果为true,说明errorString就是error的子类型。

创建标准错误

func f1(arg int) (int, error) {
    if arg == 42 {
        return -1, errors.New("can't work with 42")  // 使用错误提示创建标准错误
    }
    return arg + 3, nil
}


type argError struct {
    arg  int
    prob string
}


func (e *argError) Error() string {  // 自定义错误类型需要实现Error函数
    return fmt.Sprintf("%d - %s", e.arg, e.prob)
}
func f2(arg int) (int, error) {
    if arg == 42 {
        return -1, &argError{arg, "can't work with it"}
    }
    return arg + 3, nil
}
func main() {
    for _, i := range []int{7, 42} {
        if r, e := f1(i); e != nil {
            fmt.Println("f1 failed:", e)  // f1 failed: can't work with 42
        } else {
            fmt.Println("f1 worked:", r)
        }
    }
    for _, i := range []int{7, 42} {
        if r, e := f2(i); e != nil {
            fmt.Println("f2 failed:", e)
        } else {
            fmt.Println("f2 worked:", r)
        }
    }
    _, e := f2(42)
    if ae, ok := e.(*argError); ok {
        fmt.Println(ae.arg)
        fmt.Println(ae.prob)
    }
}

格式化消息的标准错误类型

fmt包中有个Errorf方法,我们可以通过它创建一个具有格式化字符串的标准错误类型,来看一下fmt.Errorf的源码:
func Errorf(format string, a ...interface{}) error {
  p := newPrinter()
  p.wrapErrs = true
  p.doPrintf(format, a)
  s := string(p.buf)
  var err error
  if p.wrappedErr == nil {
    err = errors.New(s)
  } else {
    err = &wrapError{s, p.wrappedErr}
  }
  p.free()
  return err
}

可以看到,fmt.Errorf里面使用errors.New来创建error类型的实例,然后把格式化后字符串放入这个实例里面。
典型使用案例:

const name, id = "bimmler", 17
err := fmt.Errorf("user %q (id %d) not found", name, id)  // user="bimmler" (id=17) not found
if err != nil {
  fmt.Print(err)
}

这样,err不仅包含了格式化的错误消息,而且还是error类型,比我们先用fmt.Printf再用errors.New的写法会更简洁。

天杀错误判断和处理逻辑

刚从其他语言转到Go语言时,会非常不习惯Go语言的错误处理逻辑。在Java和Python里面,我们可以使用throw来抛出异常,在另一个地方使用try...catch来捕获异常。而Go语言鼓励大家把错误放到返回值里面,但是呢,又提供了panic和recover来抛出和捕获异常。一般来说,我们把返回值里面返回的叫做错误(因为他们都是error类型),把抛出和捕获的叫做异常。

既然把错误放到返回值里,那我们就每次都需要判断返回里的错误是否为空,才能知道是不是正常返回了。判断和处理错误需要用到if,如果一段代码里面调用了很多函数,我们就能看到一连串的if,这种写法感觉简直就是疯了!

result1, err := err1()
if err != nil {
  fmt.Println(result1)
}
result2, err := err2()
if err != nil {
  fmt.Println(result2)
}
result3, err := err3()
if err != nil {
  fmt.Println(result3)
}
result4, err := err4()
if err != nil {
  fmt.Println(result4)
}

不过这个没法避免,老老实实习惯一下吧!


panic/recover/defer

对于异常,Go语言可以使用panic来抛出一个恐慌性异常,一般来说,当程序遇到了比较大的问题,没有办法再执行下去,我们就用panic来抛出异常。
来看一下panic源码:

func panic(v interface{})

panic接收一个interface{}类型的对象,我们上篇文章说过了,任何类型都是interface{}类型的子类型,因此,panic可以抛出任何对象。
使用案例:

func main() {
  panic("hello")
}

输出:

panic: hello

goroutine 1 [running]:
main.main()
    E:/goworkspace/blog/src/main/main.go:4 +0x40

同时,我们可以使用recover来捕获那些被panic抛出对象:

func pa() string {
  panic("me")
}
func main() {
  p := pa()
fmt.Println(p)
if r := recover(); r != nil {
   fmt.Println(r)
  }
}

输出:

panic: me

goroutine 1 [running]:
main.pa(...)
    E:/goworkspace/blog/src/main/main.go:6
main.main()
    E:/goworkspace/blog/src/main/main.go:9 +0x40
怎么好像recover没起作用?这当然了,recover需要与defer一起使用,而且,recover的使用要放在panic抛出之前。下面的例子才对:
func pa() string {
  panic("me")
}
func main() {
  defer func() {
    if r := recover(); r != nil {
      fmt.Println(r)  // 输出:me
    }
  }()
  p := pa()  // 此行panic抛出me字符串
  fmt.Println(p)  // 此行不会执行
}

defer的用法

defer关键字表示,当前这个函数在退出之前,执行一下defer后面的逻辑。相当于我们委托一些操作给Go语言,在函数退出之前执行,有点像Java和Python里面的finally作用了。

这样,我们就可以把一些一定要执行的操作,放在defer中了,比如,我们关闭某个已经打开的资源:

fsrc, err := os.Open("source.txt")
if err != nil {
  fmt.Println("open source file failed")
  return
}
defer fsrc.Close()  // 第一个defer
fdes, err := os.Open("target.txt")
if err != nil {
  fmt.Println("open target file failed")
  return
}
defer fdes.Close()  // 第二个defer
fmt.Println("do something here")

如果一个函数里面有多个defer,那么在函数推出之前,会先执行最后一个defer,然后再执行倒数第二个、倒数第三个...,就像后进先出栈的操作。

我们下一篇聊聊通道和goroutine。
 
喜欢的可以关注我的WeiXin订阅号:程序猿架构
 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
第01章 课程介绍 1-1 导学.mp4 1-2 课程介绍.mp4 第02章 实战-“云存储”系统原型 2-1 “云存储”系统原型之简单文件上传服务架构说明.mp4 2-2 编码实战:“云存储”系统之实现上传接口.mp4 2-3 编码实战:“云存储”系统之保存文件元信息.mp4 2-4 编码实战:“云存储‘系统之实现单个文件查询信息接口.mp4 2-5 编码实战:“云存储”系统之实现文件下载接口.mp4 2-6 编码实战:“云存储”系统之实现文件修改接口+小结.mp4 第03章 “云存储”系统之基于MySQL实现的文件数据库 3-1 MySQL基础知识.mp4 3-2 MySQL主从数据同步演示.mp4 3-3 文件表的设计及创建.mp4 3-4 编码实战:“云存储”系统之持久化元数据到文件表.mp4 3-5 编码实战:“云存储”系统之从文件表中获取元数据.mp4 3-6 Docker入门基础文档.mp4 3-6 本章小结.mp4 3-7 Ubuntu中通过Docker安装配置MySQL主从节点.mp4 第04章 “云存储”系统之基于用户系统实现的资源隔离及鉴权 4-1 帐号系统介绍与用户表设计.mp4 4-2 编码实战:“云存储”系统之实现用户注册接口.mp4 4-3 编码实战:“云存储”系统之实现用户登录接口.mp4 4-4 编码实战:“云存储”系统之实现用户信息查询接口.mp4 4-5 接口梳理小结.mp4 4-6 编码实战:“云存储”系统之快速实现访问鉴权接口+小结.mp4 4-7 关于静态资源访问404的问题【补漏.mp4 第05章 “云存储”系统之基于Hash计算实现秒传 5-1 Hash算法对比及秒传原理.mp4 5-2 用户文件表设计与创建.mp4 5-3 编码实战:“云存储”系统之升级改造上传接口.mp4 5-4 编码实战:“云存储”系统之基于用户查询文件Hash信息.mp4 5-5 编码实战:“云存储”系统之实现秒传功能接口+小结.mp4 第06章 “云存储”系统之基于Redis实现分块上传及断点续传 6-1_分块上传与断点续传原理.mp4 6-2_编码实战:Go实现Redis连接池(存储分块信息).mp4 6-3_编码实战:实现初始化分块上传接口.mp4 6-4_编码实战:实现分块上传接口.mp4 6-5_编码实战:实现分块合并接口.mp4 6-6_分块上传场景测试+小结.mp4 6-7_文件断点下载原理.mp4 第07章 “云存储”系统之基于Ceph实现私有云存储服务 7-1_Ceph是什么.mp4 7-2_Ceph集群介绍及兼容亚马逊S3接口详解.mp4 7-3_编码实战:Go访问管理Ceph集群.mp4 7-4_编码实战:Go实现Ceph的文件上传下载+小结.mp4 7-5_Ubuntu下通过Docker快速搭建Ceph测试集群(单机部署).mp4 7-6_Centos7下Docker部署Ceph集群(nautilus最新版,多机部署).mp4 第08章 “云存储”系统之基于阿里云OSS实现海量数据上云 8-1_阿里云对象存储OSS简介.mp4 8-2_阿里云对象存储OSS特点.mp4 8-3_阿里云对象存储OSS专业术语.mp4 8-4_阿里云对象存储OSS控制台管理.mp4 8-5_编码实战:OSS上传文件.mp4 8-6_编码实战:OSS下载文件.mp4 8-7_编码实战:OSS对象生命周期管理等常用功能.mp4 8-8_阿里云OSS本章小结.mp4 第09章 “云存储”系统之基于RabbitMQ实现异步存储 9-1_Ubuntu下通过Docker安装RabbitMQ.mp4 9-2_关于任务的同步与异步.mp4 9-3_RabbitMQ简介.mp4 9-4_RabbitMQ工作原理和转发模式.mp4 9-5_Docker安装RabbitMQ及UI管理.mp4 9-6_编码实战_实现异步转移的MQ生产者.mp4 9-7_编码实战_实现异步转移的MQ消费者.mp4 9-8_编码实战_异步转移文件测试+小结.mp4 第10章 “云存储”系统之架构微服务化 10-1_基于Docker部署服务注册发现中心consul集群.mp4 10-2_微服务基础概念与原理.mp4 10-3_云存储系统之微服务架构(1).mp4 10-4_云存储系统之微服务架构(2).mp4 10-5_Web框架Gin基础介绍.mp4 10-6_编码实战_基于Gin改造用户service(1).mp4 10-7_编码实战_基于Gin改造用户service(2.mp4 10-8_gRPC与Protobuf基础原理.mp4 10-9_RPC框架go-micro基础介绍.mp4 10-10_编码实战_改造账号系统service.mp4 10-11_编码实战_改造api网关service.mp4 10-12_编码实战_改造文件上传service.mp4 10-13_综合测试演示+小结.mp4 第11章 “云存储”系统之k8s&Docker;容器化实战 11-1_Ubuntu18下通过kubeadm单机安装k8s(v1.14.1)集群.mp4 11-2_Ubuntu18下安装k8s(v1.14.1)可视化管理工具.mp4 11-3_Docker与Docker-Compose基础概念.mp4 11-4_基于容器的微服务反向代理利器Traefik.mp4 11-5_基于Docker-compose与Traefik的容器化部署演示.mp4 11-6_Kubernetes基础原理.mp4 11-7_基于Kubernetes的容器化部署演示.mp4 第12章 “云存储”系统之持续集成部署 12-1_ubuntu下离线安装harbor1.6.mp4 12-2_持续构建之基础概念.mp4 12-3_基于gitlab+jenkins+harbor的自动化部署配置演示.mp4 第13章 课程总结 13-1_课程总结之章节重点及技能树温习.mp4

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值