首页 > 代码库 > <<Python基础教程>>学习笔记 | 第08章 | 异常
<<Python基础教程>>学习笔记 | 第08章 | 异常
------
什么是异常:Python用异常对象(exception object)来表示异常情况.如果异常信息未被处理或捕捉。
程序就会用回潄来终止执行
>>> 1/0 Traceback (most recent call last): #Traceback: 一种错误信息 File "<stdin>", line 1, in ? ZeroDivisionError: integer division or modulo by zero每个异常都是一些类的实例,这些实例可以被引发,并且可以用很多方式去捕捉,使得程序可以抓住错误并对其进行处理,而不是让整个程序失败.
Note:ZeroDivisionError:就是一个实例
------
按照自己的方式出错
raise语句:
为了引发异常,可以使用一个类或者实例参数调用raise语句。使用类时,程序会自动创建实例,下面是一些简单的例子,使用了内建的Exception异常类
#引发一个没有任何信息的普通异常
>>> raise Exception Traceback (most recent call last): File "<stdin>", line 1, in ? Exception#引发一个带错误信息的普通异常
>>> raise Exception,"This is a normal Exception" Traceback (most recent call last): File "<stdin>", line 1, in ? Exception: This is a normal Exception#也可以这么写
>>> raise Exception('System device Busy...') Traceback (most recent call last): File "<stdin>", line 1, in ? Exception: System device Busy...内建的异常很多,大部分可以再exceptions模块里面找到,可以用dir来列出模块的内容
>>> dir(exceptions) ['ArithmeticError', 'AssertionError', 'AttributeError', 'DeprecationWarning', 'EOFError', 'EnvironmentError', 'Exception', 'FloatingPointError', 'FutureWarning', 'IOError', 'ImportError', 'IndentationError', 'IndexError', 'KeyError', 'KeyboardInterrupt', 'LookupError', 'MemoryError', 'NameError', 'NotImplementedError', 'OSError', 'OverflowError', 'OverflowWarning', 'PendingDeprecationWarning', 'ReferenceError', 'RuntimeError', 'RuntimeWarning', 'StandardError', 'StopIteration', 'SyntaxError', 'SyntaxWarning', 'SystemError', 'SystemExit', 'TabError', 'TypeError', 'UnboundLocalError', 'UnicodeDecodeError', 'UnicodeEncodeError', 'UnicodeError', 'UnicodeTranslateError', 'UserWarning', 'ValueError', 'Warning', 'ZeroDivisionError', '__doc__', '__name__']Note:
1.所有这些异常都可以用在raise语句中.
2. 最重要的一些内建异常类
------
自定义异常类:
如果内建的异常类不能满足需要,那么就需要独立于exceptions外的异常类,也就是自定义类.
可以添加异常类,像下面的模式
>>> class customerException(Exception): pass#可以往里面添加方法
------
捕捉异常
异常最有用的地方是能被捕捉,可以用try/except来实现,比如说很两个数相除
>>> x = input('Enter the first number:') Enter the first number:10 >>> y = input('Enter the second number:') Enter the second number:0 >>> print x/y Traceback (most recent call last): File "<stdin>", line 1, in ? ZeroDivisionError: integer division or modulo by zero.为了捕捉异常并做出一些处理,可以这样重写程序
>>> try: ... x = input('Enter the 1st number:') ... y = input('Enter the 2nd number:') print x/y ... except ZeroDivisionError: ... print "The second number can't be zero!" ... Enter the 1st number:10 Enter the 2nd number:0 The second number can't be zero!Note: 如果没有捕捉异常,它就会被传播到调用它的函数中,如果依然没有捕获,这些异常就会浮到程序的最顶层,
也就是说你可以捕捉到在其他人的函数所引发的异常。
------
看,没有参数
如果捕捉到异常,又想重新引发它,那么就可以调用不带参数的raise,举个例子吧,看这么做,多有用!考虑一个能屏蔽ZeroDivisionError的计算器类,如果这个错误被激活,计算器就会打印出错误,而不是让异常传播,如果这是在用户交互的过程中使用,这就非常有用,但是,如果在程序内部使用,引发异常会更好些,那么屏蔽机制就可以关掉。下面是这样一个类的代码
class MuffledCalculator: muffled = False def cal(self,expr): try: return eval(expr) except ZeroDivisionError: if self.muffled: print 'Division by zero is illegal' else: raise
输出结果
>>> c = MuffledCalculator() >>> c.cal('10/2') 5 >>> c.cal('10/0') Traceback (most recent call last): File "<pyshell#2>", line 1, in <module> c.cal('10/0') File "D:\Learn\Python\Exception.py", line 13, in cal return eval(expr) File "<string>", line 1, in <module> ZeroDivisionError: integer division or modulo by zero >>> c.muffled = True >>> c.cal('10/0') Division by zero is illegal------
多个except语句
>>> x = input('Enter the 1st number:') Enter the 1st number:10 >>> y = input('Enter the 2nd number:') Enter the 2nd number:'hello,world!' >>> >>> print x/y Traceback (most recent call last): File "<stdin>", line 1, in ? TypeError: unsupported operand type(s) for /: 'int' and 'str'那么应该这么写:
try: x = input('Enter the 1st number:') y = input('Enter the 2nd number:') print x/y except ZeroDivisionError: print "The second number can't be zero!" except TypeError: print "That wasn't a number, was it?"------
用一个快来捕捉多个异常
try: x = input('Enter the 1st number:') y = input('Enter the 2nd number:') print x/y #拿所有可能的错误,放在元祖里面, #注意,一定要用(),否则可能出现不可预知的错误 except (ZeroDivisionError,TypeError,NameError): print "Your numbers were bogus..."------
捕捉对象
如果希望在except子句中访问异常对象本身,可以使用两个参数。比如,如果想让程序继续运行,但是又因为某种原因想记录下错误(比如只是打印给用户看),这个功能就很有用。下面的例子可以打印异常,但程序会继续运行。
try: x = input('Enter the 1st number:') y = input('Enter the 2nd number:') print x/y except (ZeroDivisionError,TypeError,NameError), e: print e print 'Hello,World!'输出结果:
>>> Enter the 1st number:10 Enter the 2nd number:0 integer division or modulo by zero 'Hello,World!' >>> Enter the 1st number:10 Enter the 2nd number:hello name 'hello' is not defined 'Hello,World!'Note: Python3.0中表示方式是except (ZeroDivisionError,TypeError,NameError) as e
------
真正的全捕捉:
就算程序能处理好几个异常,但有些异常还是会从眼皮底下溜走,还是刚才那个除法的程序,如果什么都不输入呢?
Traceback (most recent call last): File "<pyshell#0>", line 2, in <module> x = input('Enter the 1st number:') File "<string>", line 0 ^ SyntaxError: unexpected EOF while parsing这个异常逃过了try/except,而且程序员也不可能预测所有可能的异常,因此不能捕获所有异常,如果想让程序捕获所有异常,可以这样写:
try: x = input('Enter the 1st number:') y = input('Enter the 2nd number:') print x/y except: print "Something Wrong Happen!"这样就可以捕获所有异常
WRAN: 像这样捕捉所有异常是危险的,因为它会隐藏所有程序员未想到并且未做好准备处理的错误。它同样会捕捉用户终止执行CTRL+C企图,以及用sys.exit函数终止程序的企图,等等。这是用except Exception,e会更好些,或者对异常对象e进行一些检查。
------
万事大吉:
有些时候一些坏事发生时执行一段代码会很有用,这时候可以像条件,循环一样,加个else语句
>>> try: ... print "A Simple" ... except: ... print "What? Something went wrong?" ... else: ... print "Ah...Runing as planed." ... A Simple Ah...Runing as planed.使用else子句可以实现之前提到的循环
while True: try: x = input('Enter 1st number:') y = input('Enter 2nd number:') value = http://www.mamicode.com/x/y>输出结果:>>> Enter 1st number:10 Enter 2nd number:0 Invalid input, please try again! Enter 1st number:10 Enter 2nd number:e Invalid input, please try again! Enter 1st number:10 Enter 2nd number:2 x/y is: 5考虑到捕捉全部的异常有风险,可以用Exception,e来替代,并打印错误信息,程序再改进下:while True: try: x = input('Enter 1st number:') y = input('Enter 2nd number:') value = http://www.mamicode.com/x/y>输出结果:Enter 1st number:10 Enter 2nd number:0 Invalid input integer division or modulo by zero please try again! Enter 1st number:10 Enter 2nd number:hello Invalid input name 'hello' is not defined please try again! Enter 1st number:10 Enter 2nd number:2 x/y is: 5------最后
最后是finally子句,她可以用在可能的异常清理,也可以用在关闭数据库链接,关闭文件句柄等。一般是try/finally联合使用
x = None try: x = 1/0 finally: print "Cleaning Up..." del x上面的代码中,finally子句肯定会被执行,不管try子句中是否发生异常(在try子句之前初始化x的原因是如果不这样做,由于ZeroDivisionError的存在,x就永远不会被赋值。这样就会导致在finally子句中使用del删除它的时候产生异常,而且这个异常是无法捕捉的)运行这段代码,在程序崩溃之前,对于变量x的清理就完成了。
输出结果:
>>> Cleaning Up... Traceback (most recent call last): File "D:\Learn\Python\Exception.py", line 72, in <module> x = 1/0 ZeroDivisionError: integer division or modulo by zero因为使用del删除一个变量时很不负责的清理手段,所以finally子句用于关闭数据库链接,关闭文件句柄会非常有用。也可以try,except,else,finally联合使用(或者至少使用其中3个)try: 1/0 except Exception, e: print e else: print "That's well well." finally: print "Clean Up..."输出结果:>>> integer division or modulo by zero Clean Up...------异常和函数
异常和函数能很自然的工作。如果异常在函数内引发而不被处理,她就会传播至函数调用的地方。如果在哪里也没有处理异常,她会继续传播,一直到达主程序。如果那里也没有异常处理程序,程序会带着堆栈跟踪终止。
>>> def faulty(): ... raise Exception('Something is wrong.') ... >>> def ignore_exception(): ... faulty() ... >>> def handle_exception(): ... try: ... faulty() ... except: ... print 'Exception handled' ...输出结果:>>> ignore_exception() Traceback (most recent call last): File "<stdin>", line 1, in ? File "<stdin>", line 2, in ignore_exception File "<stdin>", line 2, in faulty Exception: Something is wrong. >>> handle_exception() Exception handled可以看出,faulty()产生的异常通过faulty和ignore_exception()传播,最终导致了堆栈跟踪。同样的,她也传播到了handle_exception(),但是也个函数被try/except处理了。
------
异常之禅
有些时候if/else实现会比try/except实现起来更好,让我们看看下面几个例子:
def describePerson(person): print 'Description of',person['name'] print 'Age:', person['age'] if 'occupation' in person: print 'Occupation:',person['occupation'] >>> d = {'name':'Alice','age':28} >>> describePerson(d) Description of Alice Age: 28 >>> d = {'name':'Alice','age':28,'occupation':'IT'} >>> describePerson(d) Description of Alice Age: 28 Occupation: IT代码非常直观,但是效率不高,程序会两次查找occupation一次是查看键是否存在,另外一次是获取值
def describePerson(person): print 'Description of',person['name'] print 'Age:', person['age'] try: print 'Occupation:'+ person['occupation'] except KeyError: passNOTE:这里打印职业时用的是+,而不是,否则字符串在引发异常时就会被输出。这样写直接假定occupation已经存在,如果不存在的话,在except KeyError中捕获.从而提高了程序的效率.在查看对象是否存在特定值时,try/except也很有用try: obj.write except AttributeError: print 'The object is not writeable' else: print 'The object is writeable.'------本章函数
warning.filterwarnings(action) 用于过滤警告
<<Python基础教程>>学习笔记 | 第08章 | 异常