在程序运行过程中,总会遇到各种各样的错误。

有的错误是程序编写有问题造成的,比如本来应该输出整数结果输出了字符串,这种错误我们通常称之为bug,bug是必须修复的。

有的错误是用户输入造成的,比如让用户输入email地址,结果得到一个空字符串,这种错误可以通过检查用户输入来做相应的处理。

还有一类错误是完全无法在程序运行过程中预测的,比如写入文件的时候,磁盘满了,写不进去了,或者从网络抓取数据,网络突然断掉了。这类错误也称为异常,在程序中通常是必须处理的,否则,程序会因为各种问题终止并退出。

错误处理

try...except...finally

当我们认为某些代码可能会出错时,就可以用try来运行这段代码,如果执行出错,则后续代码不会继续执行,而是直接跳转至错误处理代码,即except语句块,执行完except后,如果有finally语句块,则执行finally语句块,至此,执行完毕。

 #!/usr/bin/python3

 try:
r = 10 / 0
print ('result:',r)
except ZeroDivisionError as e:
print ('除数不能为零 ',e)
finally:
print ('--------')
print ('END')

如果没有错误发生,可以在except语句块后面加一个else,当没有错误发生时,会自动执行else语句:

 #!/usr/bin/python3

 try:
r = 10 / int('')
print ('result:',r)
except ZeroDivisionError as e:
print ('除数不能为零 ', e)
except ValueError as e: #int()可能抛出ValueError
print ('ValueError', e)
else:
print ('No Error!')
finally:
print ('--------')
print ('END')

Python的错误其实也是class,所有的错误类型都继承自BaseException,所以在使用except时需要注意的是,它不但捕获该类型的错误,还把其子类也“一网打尽”。

常见的错误类型和继承关系  https://docs.python.org/3/library/exceptions.html#exception-hierarchy

使用try...except捕获错误还有一个巨大的好处,就是可以跨越多层调用,比如函数main()调用foo()foo()调用bar(),结果bar()出错了,这时,只要main()捕获到了,就可以处理:

 #!/usr/bin/python3

 def foo(s):
return 10 / s def bar(s):
return foo(s)*2 def main():
try:
bar('')
except Exception as e:
print ('Error', e)
finally:
print ('---------------') main()

Python中的异常分类

 NameError    #尝试访问一个未声明的变量(没有初始化)
ZeroDivisionError #除数为零
SyntaxError #解释器语法错误,唯一不是在运行时发生的异常,在编译时发生,它代表Python代码中有一个不正确的结构
IndexError #请求的索引超出序列范围
KeyError #请求一个不存在的字典键
IOError #输入输出错误:(1)打开一个不存在的文件(2)文件写入时,操作和模式不匹配(3)权限问题导致
TypeError #类型错误,如'2'+2,类型不相同不能相加
AttributeError #尝试访问未知的对象属性,即对象根本没有定义这一属性却尝试访问它
StopIteration #迭代器没有更多的值
RuntimeError #一般的运行时错误
SystemError #一般的解释器系统错误
ValueError #传入无效的参数

调用栈

如果错误没有被捕获,它就会一直往上抛,最后被Python解释器捕获,打印一个错误信息,然后程序退出。

 #!/usr/bin/python3

 def foo(s):
return 10 / int(s) def bar(s):
return foo(s)*2 def main():
bar('') main()

可以看到,问题的源头追溯到第四行代码,第四行代码,除数不能为零

记录错误

捕获错误,打印错误堆栈,同时让程序继续执行

 #!/usr/bin/python3

 import logging

 def foo(s):
return 10 / int(s) def bar(s):
return foo(s)*2 def main():
try:
bar('')
except Exception as e:
logging.exception(e) main()
print('END')

同样是出错,但程序打印完错误信息后会继续执行,并正常退出:

抛出错误

如果要抛出错误,首先根据需要,可以定义一个错误的class,选择好继承关系,然后,用raise语句抛出一个错误的实例:

 #!/usr/bin/python3

 class FooError(ValueError):   #选择好继承关系
pass def foo(s):
n = int(s)
if n == 0:
raise FooError('无效的值:%s' % s)
return 10 / n foo('')

只有在必要的时候才定义我们自己的错误类型。如果可以选择Python已有的内置的错误类型(比如ValueErrorTypeError),尽量使用Python内置的错误类型。

调试

程序能一次写完并正常运行的概率很小,总会有各种各样的bug需要修正。有的bug很简单,看看错误信息就知道,有的bug很复杂,我们需要知道出错时,哪些变量的值是正确的,哪些变量的值是错误的,因此,需要一整套调试程序的手段来修复bug。

断言   assert

把可能有问题的变量打印出来看看

 #!/usr/bin/python3

 def foo(s):
n = int(s)
assert n != 0, 'n is zero!'
return 10 / n def main():
foo('') main()

assert的意思是,表达式 n != 0应该是True,True的话继续执行,否则,根据程序运行的逻辑,后面的代码肯定会出错。

如果断言失败,assert语句本身就会抛出AssertionError

logging

assert比,logging不会抛出错误,而且可以输出到文件,可指定记录信息的级别(低到高:debug  info  warning  error)当指定 level = WARNING后 debug和info就不起作用,即可以输出不同级别的信息

 #!/usr/bin/python3

 import logging
logging.basicConfig(level = logging.INFO) s = ''
n = int(s)
logging.info('n = %d' % n)
print (10 / n)

pdb

-m pdb  让程序以单步方式运行,可以随时查看运行状态。

 #!/usr/bin/python3

 s = ''
n = int(s)
print (10 / n)

l  查看代码  n  单步  p 变量名    查看变量  q结束调试

python错误和调试的更多相关文章

  1. Python错误、调试和测试

    try: print('try...') r = 10 / int('a') print('result:', r) except ValueError as e: print('ValueError ...

  2. python 错误、调试和测试

    在程序运行过程中,总会遇到各种各样的错误. 有的错误是程序编写有问题造成的,比如本来应该输出整数结果输出了字符串,这种错误我们通常称之为bug,bug是必须修复的. 有的错误是用户输入造成的,比如让用 ...

  3. python错误处理/调试/单元测试/文档测试

    一.错误处理 1.错误处理 try: ... except Exception1: ... except Exception2: ... finally: ... 如果在try中发生错误,那么exce ...

  4. python 错误、调试、单元测试、文档测试

    错误分为程序的错误和由用户错误的输入引起的错误,此外还有因为各种各样意外的情况导致的错误,比如在磁盘满的时候写入.从网络爬取东西的时候,网络断了.这类错误称为异常 错误处理 参考链接:https:// ...

  5. python错误、调试、测试

    1.错误, Python内置的try...except...finally用来处理错误十分方便.出错时,会分析错误信息并定位错误发生的代码位置才是最关键的. 程序也可以主动抛出错误,让调用者来处理相应 ...

  6. Python:笔记(5)——错误、调试和测试

    Python:笔记(5)——错误.调试和测试 错误处理 1.TRY语句 这个和Java中的语法是及其相似的,catach换成except. 说明:同样,不管有没有错误,fianlly都会执行的! 补充 ...

  7. python: 错误处理try详解 ,traceback调用栈, 调试(logging)

    摘录:https://www.liaoxuefeng.com/wiki/1016959663602400/1017598873256736 错误处理 调试 错误处理 高级语言都会使用内置的一套try. ...

  8. [转] python程序的调试方法

    qi09 原文 python程序的调试方法 本文讨论在没有方便的IDE工具可用的情况下,使用pdb调试python程序 源码例子 例如,有模拟税收计算的程序: #!/usr/bin/python de ...

  9. 004-python面向对象,错误,调试和测试

    ---恢复内容开始--- 1.面向对象 面向对象编程——Object Oriented Programming,简称OOP,是一种程序设计思想.OOP把对象作为程序的基本单元,一个对象包含了数据和操作 ...

随机推荐

  1. spring security使用自定义登录界面后,不能返回到之前的请求界面的问题

    昨天因为集成spring security oauth2,所以对之前spring security的配置进行了一些修改,然后就导致登录后不能正确跳转回被拦截的页面,而是返回到localhost根目录. ...

  2. 第18章 启动 - Identity Server 4 中文文档(v1.0.0)

    IdentityServer是中间件和服务的组合.所有配置都在您的启动类中完成. 18.1 配置服务 您可以通过调用以下方法将IdentityServer服务添加到DI系统: public void ...

  3. [转]gitlab配置通过smtp发送邮件(QQ exmail腾讯企业为例)

    本文转自:http://www.fayfox.com/post/39.html 首先祭出官网文档链接:https://docs.gitlab.com/omnibus/settings/smtp.htm ...

  4. C# 程序异常关闭时的捕获

    本文主要以一个简单的小例子,描述C# Winform程序异常关闭时,如何进行捕获,并记录日志. 概述 有时在界面的事件中,明明有try... catch 进行捕获异常,但是还是会有异常关闭的情况,所以 ...

  5. 关于我的博客(About My Blogs)

    本文算是第一篇文章,记录工作/学习/生活中的觉得值得分享的东西 When,Where,Who,What,How,Why,尽可能每篇博文当作一个事件来描述. 也就是触发想写这篇文章的事件. 要坚持下来写 ...

  6. csharp: LocalDataCache.sync

    app.config: <?xml version="1.0" encoding="utf-8" ?> <configuration> ...

  7. IEC104协议规约解析

    一.四遥信息体基地址范围 104调度规约有1997年和2002年两个版本,在流程上没有什么变化,02版只是在97版上扩展了遥测.遥信等信息体基体址,区别如下: 类别 1997版基地址 2002版基地址 ...

  8. python读取txt文件最后一行(文件大+文件小)

    txt文件小 #coding:utf-8 ''' fname为所读xx.txt文件 输出为:文件第一行和最后一行 ''' fname = 'test.txt' with open(fname, 'r' ...

  9. 一个表里有多个字段需要同时使用字典表进行关联显示,如何写sql查询语句

    参考:https://bbs.csdn.net/topics/330032307 数据库里面有一个字典表,这张表里面有id段和对应的名字字段.在另外一个记录的表里面有对应的上述字典表的id,而且有多个 ...

  10. 《我们不一样》Alpha冲刺_1-5

    第一天    日期:2018/6/15 1.1 今日完成任务情况以及遇到的问题. 马    兰.马   娟:用户.管理员数据库表的设计 李国栋.张惠惠:前端登录界面代码书写 伊力亚.张   康:配置s ...