Go语言在循环中panic后还能继续执行for循环?

Python020

Go语言在循环中panic后还能继续执行for循环?,第1张

如下的例子,要打印100以内能被5整除的数,以panic的方式选择出来并打印。

如果用下面的方式,执行到第一个panic就会跳出for循环

只能输出第一个匹配项,然后退出for循环。

那么如何保证在for循环处理完panic不退出循环,直到打印完所有满足条件的数值?

golang的panic被恢复后,能继续执行比recover更早的defer,或者返回到recover函数的调用方,然后继续执行下去。

所以,我们可以把panic和recover放到单独的函数中,然后在for循环里调用这个函数,这个函数panic并恢复后,能返回到调用方for循环并继续循环下去。

执行结果是所有0到100的所有符合panic条件的都能正确处理,for循环没有异常退出:

golang的panic属于非常严重的错误,一旦panic没有recover的话,程序就退出了。一般避免主动panic,影响程序稳定性。

recover函数要放在defer里面,并且只能恢复同一个goroutine的并且是直接调用链函数发生的panic。recover不能恢复上一层函数的panic。

panic的作用就是抛出一条错误信息,从它的参数类型可以看到它可以抛出任意类型的错误信息。在函数执行过程中的某处调用了panic,则立即抛出一个错误信息,同时函数的正常执行流程终止,但是该函数中panic之前定义的defer语句将被依次执行。之后该goroutine立即停止执行。

recover()用于将panic的信息捕捉。recover必须定义在panic之前的defer语句中。在这种情况下,当panic被触发时,该goroutine不会简单的终止,而是会执行在它之前定义的defer语句。

建议每起一个goroutine,都defer统一捕获下异常

谨记用defer统一捕获异常只对当前的goroutine有效,goroutine的异常并不会向上传递给main主函数

golang中defer,panic,recover是很常用的三个特性,三者一起使用可以充当其他语言中try…catch…的角色,而defer本身又像其他语言的析构函数

结果:

例1

例2

例3

请先不要向下看,在心里跑一遍上边三个例子的结果,然后去验证

可能你会认为: 例1的结果是0,例2的结果是10,例3的结果是6 ,那么很遗憾的告诉你,这三个结果都错了

为什么呢,最重要的一点就是要明白, return xxx这一条语句并不是一条原子指令

含有defer函数的外层函数,返回的过程是这样的:先给返回值赋值,然后调用defer函数,最后才是返回到更上一级调用函数中,可以用一个简单的转换规则将return xxx改写成

例1可以改写成这样

所以例1的返回值是1

例2可以改写成这样

所以例2的结果是5

例3可以改写成这样

所以例3的结果是1

在defer函数申明时,对外部变量的引用是有两种方式:作为函数参数和作为闭包引用

作为函数参数,在defer申明时就把值传递给defer,并将值缓存起来,调用defer的时候使用缓存的值进行计算(如上边的例3)

而作为闭包引用,在defer函数执行时根据整个上下文确定当前的值

看个