python asyncio 异常处理_Python异常的处理记录

参考资料来至:PythonCookbook

首先要知道,所有的异常事一个类。

如果一个单独的代码块处理所有不同的异常,可以将它们归组到一个元祖中。

from urllib.error import URLError

try:

object.get_url(url)

except (URLError, ValueError, SocketTimeout):

object.remove_url(url)

这个错误元祖(URLError, ValueError, SocketTimeout)捕获到都会执行object.remove_url(url)

如果对一个单独的采取不同的处理方法,可以将其放入一个单独的except中

try:

object.get_url(url)

except (URLError, ValueError):

object.remove_url(url)

except SocketTimeout:

object.other_do(url)

有许多归组为继承体系,对于这样的异常,可以通过指定一个基类来捕获所有异常。

try:

f = open('a.txt')

except (FileNotFoundError, PermissionError):

...

try:

f = open('a.txt')

except OSError:

...

# 查下此类下面的子类

print(OSError.__subclasses__())

print()

# 查看继承表

print(FileNotFoundError.__mro__)

[, , , ,

, , , ,

, , , , ]

(, , , , )

书中不严谨的地方事,明显OSError包含的错误类型更加多。

我们还可以通过as e,这个e就是捕获到错误的实例,实例的话就应该有一堆属性,你也可以通过type(e)的方式查看e的类型,便于精准捕获。

其中的error属性可以捕获到错误的代码。

except向下执行捕获到合适的异常,下面的except就不会执行。

try:

f = open('a.txt')

except OSError as e:

print(dir(e))

print('OSError')

# 这个肯定不会执行,应该OSError是FileNotFoundError的父类,它把包含它的所有异常都捕获了

except FileNotFoundError:

print('FileNotFoundError')

下面上一下OSError对象的属性

['__cause__', '__class__', '__context__', '__delattr__', '__dict__',

'__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__gt__',

'__hash__', '__init__', '__init_subclass__', '__le__', '__lt__', '__ne__', '__new__',

'__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__setstate__', '__sizeof__', '__str__',

'__subclasshook__', '__suppress_context__', '__traceback__', 'args', 'characters_written', 'errno', 'filename', 'filename2', 'strerror', 'with_traceback']

捕获所有的异常

try:

...

# 一般错误这个够用了

except Exception as e:

...

# 这个是错误的祖宗

except BaseException as e:

...

除了SystemExit,KeyboardInterrupt,和GeneratorExit之外,Exception都能捕获,如果想前面几个异常也捕获可以用BaseException。

书中介绍说明,当我们捕获错误的时候,如果选择了Exception这样的广义错误,最好用as e,把e输出显示,这样再出现问题的时候,我们可以从实例e的输出中查看具体的情况。

创建自定义的异常

我们日常的使用中需要一些自己定义的异常,简单的操作,可以继承该类,并自己重新定义一个自己熟悉的类名。

实际操作中,我们一般总继承内奸的Exception类,或者继承自一些本地定义的基类,而这个基类本身又是继承自Exception的,不能去继承BaseException。

如果改写Exception的__init__方法,请确保所有参数传递给Exception的__init__,Exception默认的认为就是接收所有传递过来的参数,并将它以元祖的形式保存再args中。

In [82]: class My_Exception(Exception):

...: def __init__(self,message, status):

...: super().__init__(message, status)

...: self.message = message

...: self.status = status

...:

In [85]: try:

...: raise My_Exception('my error','888')

...: except My_Exception as e:

...: print(e.message)

...: print(e.status)

...: print(e.args)

...:

...:

...:

my error

888

('my error', '888')

通过引发异常来响应另一个异常。

要将异常串联起来,可以使用yield from。

In [94]: def example():

...: try:

...: int('n/a')

...: except ValueError as e:

...: raise RuntimeError('A parseing error occurred') from e

...:

In [95]: example()

---------------------------------------------------------------------------

ValueError Traceback (most recent call last)

in example()

2 try:

----> 3 int('n/a')

4 except ValueError as e:

ValueError: invalid literal for int() with base 10: 'n/a'

The above exception was the direct cause of the following exception:

RuntimeError Traceback (most recent call last)

in

----> 1 example()

in example()

3 int('n/a')

4 except ValueError as e:

----> 5 raise RuntimeError('A parseing error occurred') from e

6

RuntimeError: A parseing error occurred

In [96]:

这一行是关键The above exception was the direct cause of the following exception:

在生成器throw(StopItoration)里面引起RuntimeError应该也是采用了yield from的方式。

In [97]: try:

...: example()

...: except RuntimeError as e:

...: print('It didnot work', e)

...: if e.__cause__:

...: print('Cause', e.__cause__,'type',type(e.__cause__))

...:

...:

It didnot work A parseing error occurred

Cause invalid literal for int() with base 10: 'n/a' type

我们可以从捕获的RuntimeError对象e里面发现有__cause__属性,调用后,为引起它错误的实例。

In [98]: def example():

...: try:

...: int('n/a')

...: except ValueError:

...: raise RuntimeError('A parseing error occurred')

...:

...:

In [99]: try:

...: example()

...: except RuntimeError as e:

...: print('It didnot work', e)

...: if e.__cause__:

...: print('Cause', e.__cause__,'type',type(e.__cause__))

...:

...:

It didnot work A parseing error occurred

In [100]: example()

---------------------------------------------------------------------------

ValueError                                Traceback (most recent call last)

in example()

2     try:

----> 3         int('n/a')

4     except ValueError:

ValueError: invalid literal for int() with base 10: 'n/a'

During handling of the above exception, another exception occurred:

RuntimeError                              Traceback (most recent call last)

in

----> 1 example()

in example()

3         int('n/a')

4     except ValueError:

----> 5         raise RuntimeError('A parseing error occurred')

6

7

RuntimeError: A parseing error occurred

上面去掉了yield from 可以发现两个错误没有因果关系,调用__cause__属性为空.

在except里面无论是否是raise还是raise from,该脚本的调用者,抓取异常以最后复现的异常为准。就像前面写的函数,最后抓取函数异常,只能通过RuntimeError抓取。

如果处于某种元婴想阻止异常链的发生,可以使用raise from None 完成

In [103]: example()

---------------------------------------------------------------------------

RuntimeError Traceback (most recent call last)

in

----> 1 example()

in example()

3 int('n/a')

4 except ValueError:

----> 5 raise RuntimeError('A parseing error occurred') from None

6

7

RuntimeError: A parseing error occurred

在设计代码的时候,如果需要再except中raise 错误,经量应该使用raise from,这样能够明确的表达处你希望引发第二个异常的意图。

少出现直接except下直接raise 下一个错误。

重新抛出上一个异常

我们在except块中捕获了一个异常,现在想将它重新抛出,可以再except中重新使用raise。

In [106]: def example():

...: try:

...: int('a')

...: except ValueError as e:

...: print(e)

...: raise

...:

In [107]: example()

invalid literal for int() with base 10: 'a'

---------------------------------------------------------------------------

ValueError Traceback (most recent call last)

in

----> 1 example()

in example()

1 def example():

2 try:

----> 3 int('a')

4 except ValueError as e:

5 print(e)

ValueError: invalid literal for int() with base 10: 'a'

这个以后再工作中,对异常进行采集还是非常有用的,采集了以后再次进行上浮报错。

你可能感兴趣的:(python,asyncio,异常处理)