Python 异常

Python 含有异常处理机制来帮助用户处理可能发生的错误异常。

1. 异常概念

异常是指Python程序运行过程中遇到的错误, 每个异常都是某个异常类的实例。下面实例中未对异常做任何处理,导致程序终止并且抛出错误信息。其中TypeError是异常类中的一种。

def test(a=1, b='2'):
    return a + b

test()

#result
Traceback (most recent call last):
  File "test.py", line 4, in <module>
    test()
  File "test.py", line 2, in test
    return a + b
TypeError: unsupported operand type(s) for +: 'int' and 'str'

2. 抛出异常

Python中抛出异常必须用到raise语句, 后面接上异常类(会自动创建实例)或者异常类的实例

raise Exception
#result
Traceback (most recent call last):
  File "test.py", line 1, in <module>
    raise Exception
Exception
#############################
raise Exception()

#result
Traceback (most recent call last):
  File "test.py", line 1, in <module>
    raise Exception
Exception
#############################
raise Exception('Meet Error.')

#result
Traceback (most recent call last):
  File "test.py", line 1, in <module>
    raise Exception('Meet Error.')
Exception: Meet Error.

抛出异常的进阶:

def test(a=1, b='2'):
    return a + b

try:
    test()
except Exception as e:
    raise
    #raise RuntimeError('Raise Error.')
    #raise RuntimeError('Raise Error.') from e
    #raise RuntimeError('Raise Error.') from None

#result 1
Traceback (most recent call last):
  File "test.py", line 5, in <module>
    test()
  File "test.py", line 2, in test
    return a + b
TypeError: unsupported operand type(s) for +: 'int' and 'str'

#result 2
Traceback (most recent call last):
  File "test.py", line 5, in <module>
    test()
  File "test.py", line 2, in test
    return a + b
TypeError: unsupported operand type(s) for +: 'int' and 'str'

During handling of the above exception, another exception occurred:

Traceback (most recent call last):
  File "test.py", line 8, in <module>
    raise RuntimeError('Raise Error.')
RuntimeError: Raise Error.

#result 3
Traceback (most recent call last):
  File "test.py", line 5, in <module>
    test()
  File "test.py", line 2, in test
    return a + b
TypeError: unsupported operand type(s) for +: 'int' and 'str'

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

Traceback (most recent call last):
  File "test.py", line 9, in <module>
    raise RuntimeError('Raise Error.') from e
RuntimeError: Raise Error.

#result 4
Traceback (most recent call last):
  File "test.py", line 10, in <module>
    raise RuntimeError('Raise Error.') from None
RuntimeError: Raise Error.
  • 第一种: 直接raise, 捕获异常后,重新引发该异常
  • 第二种: 抛出RuntimeError,处理异常时发生了新的异常,更倾向于新异常与正在处理的异常没有关系
  • 第三种: from e, 指出新异常是因旧异常直接引起的, 有助于后续对异常的分析和排查
  • 第四种: from None, 明确禁止异常关联

3. 异常类

异常类分两种, 一是内置异常类,如ExceptionOSErrorSyntaxError等, 直接拿来就可以用。
BaseException是所有异常类的基类。详见内置异常

image.png

二是为满足个人需求的自定义异常类, 这种类必须要直接或间接地继承 Exception, 然后根据需求是否添加方法。

class CaseError(Exception):
   pass

raise CaseError('Case is error.')

#result
Traceback (most recent call last):
  File "test.py", line 4, in <module>
    raise CaseError('Case is error.')
__main__.CaseError: Case is error.

4. 捕获异常

可以使用try/except来处理可能发生异常的语句。

4.1 捕获一种异常

使用try/except + 异常类名来捕获该异常

def test(a=1, b='2'):
    return a + b

try:
    test()
except TypeError:
    print ('Meet type error.')

#result
Meet type error.
4.2 捕获多种异常

有时候程序往往出现不止一种异常, 那如何捕获多个异常呢,有下面的两种方法。

  • 多层次捕获: 层次分明, 针对不同的异常做不同的处理
    注意:后面第二个至第n个类中不能是第一个的子类, 否则该异常永远都无法被捕获
def test(a=1, b='2'):
    return a + b

try:
    test()
except TypeError:
    print ('It is type error.')
except NameError:
    print ('It is name error.')

#result
It is type error.
  • 一起捕获: 把所有的异常类放在一个元祖里面
def test(a=1, b='2'):
    return a + b

try:
    test()
except (TypeError,  NameError):
    print ('Meet error.')

#result
Meet error.
4.3 捕获所有异常

捕获所有异常有两种常见的格式, 分别是try/excepttry/except Exception

  • try/except 会捕获所有异常,包含键盘中断(KeyboardInterrupt)程序退出请求(SystemExit),慎用之。可能会使sys.exit() 语句被捕获,导致无法退出脚本。
import sys

try:
    sys.exit(0)
    print ('Exit the function.')
except:
    print ('Meet error.')

#result
Meet error.
# sleep 的时候,按Ctrl+c 
import time

try:
    time.sleep(20)
    print ('Sleep 20s.')
except:
    print ('Meet error.')

#result,^C表示Ctrl+c 
^CMeet Error.
  • try/except Exception 会捕获大部分异常,除了键盘中断(KeyboardInterrupt)程序退出请求(SystemExit),因为它们是从 BaseException ( Exception 的超类)派生而来的。
import sys

try:
    sys.exit(0)
    print ('Exit the function.')
except Exception:
    print ('Meet error.')
#result
退出脚本

###############
import time

try:
    time.sleep(20)
    print ('Sleep 20s.')
except Exception:
    print ('Meet error.')

#result
^CTraceback (most recent call last):
  File "test.py", line 4, in <module>
    time.sleep(20)
KeyboardInterrupt

5. 获取更多的异常信息

当捕获异常的时候, 我们需要获取更多的信息去定位和解决问题。
这里有三种方式。一是try/except XXX as e, 另外是traceback内置模块,第三是logging内置模块。

1. e表示异常对象
2. print_exc():对异常的输出
3. format_exc():把异常以字符串的形式返回,print(traceback.format_exc())等同于于traceback.print_exc()
4. print_exception():traceback.print_exc()实现方式就是traceback.print_exception(sys.exc_info())
import sys
import logging
import traceback

def test(a=1, b='2'):
    return a + b

try:
    test()
except Exception as e:
    print ('****** print object e ******')
    print ('Exception type is %s.' % type(e))
    print ('e is %s.' % e)

    print ('****** traceback.print_exc() ******')
    traceback.print_exc()

    print ('****** traceback.format_exc() ******')
    print(traceback.format_exc())

    print ('****** traceback.print_exception() ******')
    traceback.print_exception(*sys.exc_info())

    print ('****** logging_exception() ******')
    traceback.print_exception(*sys.exc_info())
###########result###########
****** print object e ******
Exception type is <class 'TypeError'>.
e is unsupported operand type(s) for +: 'int' and 'str'.
****** traceback.print_exc() ******
Traceback (most recent call last):
  File "test.py", line 8, in <module>
    test()
  File "test.py", line 5, in test
    return a + b
TypeError: unsupported operand type(s) for +: 'int' and 'str'
****** traceback.format_exc() ******
Traceback (most recent call last):
  File "test.py", line 8, in <module>
    test()
  File "test.py", line 5, in test
    return a + b
TypeError: unsupported operand type(s) for +: 'int' and 'str'

****** traceback.print_exception() ******
Traceback (most recent call last):
  File "test.py", line 8, in <module>
    test()
  File "test.py", line 5, in test
    return a + b
TypeError: unsupported operand type(s) for +: 'int' and 'str'
****** logging_exception() ******
Traceback (most recent call last):
  File "test.py", line 9, in <module>
    test()
  File "test.py", line 6, in test
    return a + b
TypeError: unsupported operand type(s) for +: 'int' and 'str'

6. else 和 finally

try/except ... else举例: else 在主 try 块没有引发异常时执行

def except_test():
    try:
        print ('Running')
        raise Exception('exception raised')
        return 0
    except Exception:
        print('Here have exception')
        return 1
    else:
        print('No exception')
        return 2

print (except_test())

#如果没有发生异常, 则返回
Running
0
#如果没有发生异常,且注释掉return 0, 则返回
Running
No exception
2
#如果发生异常,则返回
Running
Here have exception
1
#如果发生异常,且注释掉return 1, 则返回
Running
Here have exception
None

由此可见:

  • try 中有异常,会直接跳到except语句
  • try 中没有异常, 且没有return,才会跳到else语句

try/except ... else ... finally举例:无论是否引发异常都将执行

def except_test():
    try:
        print ('Running')
        raise Exception('exception raised')
        return 0
    except Exception:
        print('Here have exception')
        return 1
    else:
        print('No exception')
        return 2
    finally:
        print ('Do clean work')
        return 3

print (except_test())

#如果没有发生异常, 则返回
Running
Do clean work
3
#如果没有发生异常,且注释掉return 0, 则返回
Running
No exception
Do clean work
3
#如果没有发生异常,且注释掉return 3, 则返回
Running
Do clean work
0
#如果没有发生异常,且注释掉return 0 和 return 3, 则返回
Running
No exception
Do clean work
2
#如果发生异常, 则返回
Running
Here have exception
Do clean work
3
#如果发生异常, 且注释掉return 3,则返回
Running
Here have exception
Do clean work
1

由此可知:

  • try 中不管有没有异常, 有没有return, 最终都会运行finally, 且如果finally 中有return,则函数返回该return
  • finally return 优先级》try/except return 优先级》else return

7. 异常是向上传播的

在函数中引发异常时,异常将传播到调用函数的地方。

def err():
    raise Exception('Error.')

def test_err():
    err()

def test_err_too():
    test_err()

test_err_too()

#result
Traceback (most recent call last):
  File "test.py", line 10, in <module>
    test_err_too()
  File "test.py", line 8, in test_err_too
    test_err()
  File "test.py", line 5, in test_err
    err()
  File "test.py", line 2, in err
    raise Exception('Error.')
Exception: Error.
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,657评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,662评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,143评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,732评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,837评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,036评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,126评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,868评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,315评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,641评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,773评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,859评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,584评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,676评论 2 351

推荐阅读更多精彩内容

  • 什么是异常 本节开始介绍之前,先看看如下程序: 是不是很熟悉,这是我们前面经常看到的程序运行出现的错误。作为Pyt...
    泷汰泱阅读 442评论 0 0
  • @Author : Roger TX (425144880@qq.com) @Link : https:/...
    Roger田翔阅读 4,312评论 0 2
  • Python 异常处理 python提供了两个非常重要的功能来处理python程序在运行中出现的异常和错误。你可以...
    你的python助手阅读 291评论 0 2
  • 调试Python程序时,经常会报出一些异常,异常的原因一方面可能是写程序时由于疏忽或者考虑不全造成了错误,这时就需...
    有可能_666阅读 528评论 0 0
  • Python异常处理 异常概念: 异常:就是不正常的情况,程序开发过程中错误和BUG都是补充正常的情况 异常发生的...
    youngkun阅读 921评论 0 4