第七章 Python异常处理

发布时间:2019-09-18 07:26:13编辑:auto阅读(1602)

    什么是异常?

    顾名思义,异常就是程序因为某种原因无法正常工作了,比如缩进错误、缺少软件包、环境错误、连接超时等等都会引发异常。一个健壮的程序应该把所能预知的异常都应做相应的处理,应对一些简单的异常情况,使得更好的保证程序长时间运行。即使出了问题,也可让维护者一眼看出问题所在。因此本章节讲解的就是怎么处理异常,让你的程序更加健壮。

    7.1 捕捉异常语法

    try...except...
    try:
         expression
    except [Except Type]:
         expression

    7.2 异常类型

    常见的异常类型:

    异常类型

    用途

    SyntaxError语法错误
    IndentationError缩进错误
    TypeError  
    对象类型与要求不符合
    ImportError模块或包导入错误;一般路径或名称错误
    KeyError
    字典里面不存在的键
    NameError变量不存在
    IndexError下标超出序列范围
    IOError输入/输出异常;一般是无法打开文件
    AttributeError对象里没有属性
    KeyboardInterrupt键盘接受到Ctrl+C
    Exception通用的异常类型;一般会捕捉所有异常

    还有一些异常类型,可以通过dir查看:

    >>> import exceptions
    >>> dir(exceptions)
    ['ArithmeticError', 'AssertionError', 'AttributeError', 'BaseException', 'BufferError', 'BytesWarning', 'DeprecationWarning', 'EOFError', 'EnvironmentError', 'Exception', 'FloatingPointError', 'FutureWarning', 'GeneratorExit', 'IOError', 'ImportError', 'ImportWarning', 'IndentationError', 'IndexError', 'KeyError', 'KeyboardInterrupt', 'LookupError', 'MemoryError', 'NameError', 'NotImplementedError', 'OSError', 'OverflowError', 'PendingDeprecationWarning', 'ReferenceError', 'RuntimeError', 'RuntimeWarning', 'StandardError', 'StopIteration', 'SyntaxError', 'SyntaxWarning', 'SystemError', 'SystemExit', 'TabError', 'TypeError', 'UnboundLocalError', 'UnicodeDecodeError', 'UnicodeEncodeError', 'UnicodeError', 'UnicodeTranslateError', 'UnicodeWarning', 'UserWarning', 'ValueError', 'Warning', 'ZeroDivisionError', '__doc__', '__name__', '__package__']

    7.3 异常处理

    例如:打印一个没有定义的变量

    >>> print a
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
    NameError: name 'a' is not defined

    会抛出异常,提示名字没有定义。如果程序遇到这种情况,就会终止。

    那我们可以这样,当没有这个变量的时候就变量赋值,否则继续操作。

    >>> try:             
    ...   print a
    ... except NameError:
    ...   a = ""
    ...
    >>> a
    ''

    这样就避免了异常的发生。在开发中往往不知道什么是什么异常类型,这时就可以使用Exception类型来捕捉所有的异常:

    例如:打印一个类对象里面没有的属性

    >>> class A:
    ...   a = 1
    ...   b = 2
    ...
    >>> c = A()
    >>> try:
    ...   print c.c
    ... except Exception:
    ...   print "Error..."
    ...
    Error...

    有时也想把异常信息也打印出来,怎么做呢?

    可以把错误输出保存到一个变量中,根据上面例子来:

    >>> try:
    ...   print c.c
    ... except Exception, e:
    ...   print "Error: " + str(e)
    ...
    Error: A instance has no attribute 'c'
    # 也可以使用as关键字将错误出输出保存到变量中
    >>> try:               
    ...   print c.c         
    ... except Exception as e:
    ...   print "Error: " + str(e)         
    ...
    Error: A instance has no attribute 'c'

    当出现的异常类型有几种可能性时,可以写多个except:

    >>> try:
    ...   print a
    ... except NameError, e:
    ...   print "NameError: " + str(e)
    ... except KeyError, e:
    ...   print "KeyError: " + str(e)
    ...
    NameError: name 'a' is not defined

    注意:except也可以不指定异常类型,那么会忽略所有的异常类,这样做有风险的,它同样会捕捉Ctrl+C、sys.exit等的操作。所以使用except Exception更好些。

    7.4 else和finally语句

    7.4.1 else语句

    表示如果try中的代码没有引发异常,则会执行else。

    继续按照上面定义的类举例:

    >>> try:    
    ...   print c.a
    ... except Exception as e:
    ...   print e
    ... else:
    ...   print "else..."
    ...
    1
    else...

    7.4.2 finally语句

    表示无论是否异常,都会执行finally。

    >>> try:    
    ...   print c.c
    ... except Exception as e:
    ...   print e
    ... finally:
    ...   print "finally..."
    ...
    A instance has no attribute 'c'
    finally...

    一般用于清理工作,比如打开一个文件,不管是否文件是否操作成功,都应该关闭文件。

    7.4.3 try...except...else...finally

    这是一个完整的语句,当一起使用时,使异常处理更加灵活。

    #!/usr/bin/python    
    # -*- coding: utf-8 -*-
    try:
       print a
    except Exception as e:
       print "Error: " + str(e)
    else:
       print "else..."
    finally: 
       print "finally..."
            
    # python test.py
    python test.py
    Error: name 'a' is not defined
    finally...

    需要注意的是:它们语句的顺序必须是try...except...else...finally,否则语法错误!里面else和finally是可选的。


    博客地址:http://lizhenliang.blog.51cto.com

    QQ群:Shell/Python运维开发群 323779636


    7.5 自定义异常类

    raise语句用来手动抛出一个异常,使用方法:

    raise ExceptType(ExceptInfo)

    例如:抛出一个指定的异常

    >>> raise NameError('test except...')
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
    NameError: test except...

    raise参数必须是一个异常的实例或Exception子类。

    上面用的Exception子类,那么我定义一个异常的实例,需要继承Exception类:

    >>> class MyError(Exception):
    ...   def __init__(self, value):
    ...      self.value = value
    ...   def __str__(self):
    ...      return self.value
    ...
    >>> raise MyError("MyError...")
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
    __main__.MyError: MyError...

    7.6 assert语句

    assert语句用于检查条件表达式是否为真,不为真则触发异常。又称断言语句。

    一般用在某个条件为真才能正常工作。

    >>> assert 1==1
    >>> assert 1!=1
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
    AssertionError
    >>> assert range(4)==[0,1,2] 
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
    AssertionError
    # 添加异常描述信息
    >>> assert 1!=1, "assert description..."
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
    AssertionError: assert description...


关键字