两种编程风格:LBYL与EAFP
1. LBYL(look before you loop):
执行一个可能会出错的操作时,先做一些关键条件的判断,仅当条件满足时才进行操作。
def to_int_add(value):
"""
:param value:整型,或者可以转为整型的字符串
:return: 整型结果
"""
if isinstance(value,int):
return value+1
elif isinstance(value,str) and value.isdigit():
return int(value)+1
else:
print("error")
2. EAFP(easier to ask for forgiveness than permission)
不做任何的事前检查,直接执行操作,外层使用try来捕获可能的异常。
def to_int_add(value):
"""
:param value:整型,或者可以转为整型的字符串
:return: 整型结果
"""
try:
return int(value)+1
except (TypeError,ValueError) as e:
print(f"error:{e}")
python中抛出与捕获异常都是轻量操作,并不会给系统带来负担。
Ctrl+C
会捕获KeyboardInterrupt
异常;for循环结束
会捕获StopIteration
异常;
try语句常用知识
1. 更精确的expect语句放在前面
BaseException
是一切异常类的父类,可用issubclass(B,A)
判断B类是不是A的子类。issubclass(Exception, BaseException) #True
except BaseException
不仅捕获BaseException类异常,同时会捕获其所有子类的异常,所有异常都会被无差别捕获。
把更精确的更小范围的异常子类放在前面将会定位更准确。
2. else分支
异常捕获中的else表示:当try模块没有抛出任何异常时,才执行else分支。若捕获中断,即时没有抛出任何异常,也不会执行else分支。
3. 使用空raise语句
def to_int_add(value):
try:
return int(value)+1
except (TypeError,ValueError) as e:
print(f"error:{e}")
raise # 原封不动抛出捕获的异常
4. 使用异常来代替错误情况
对于异常情况可通过自定义异常类型来代替错误返回值。
5. 使用上下文管理器 — with
with
是一个关键字,他可以开辟一个由它管理的上下文,并控制程序进入和推出这段程序时候的行为。最常见的用法:
with open("data.txt","r") as data:
定义了一个上下文,
(1)进入时打开文件data.txt
并返回文件对象data
;
(2)退出时关闭文件该文件对象;不是所有的对象都能与with配合使用,只有满足上下文管理协议的对象才可以与with配合使用。
要自定义创建一个上下文管理器,只需要实现__enter__
和__exit__
两个方法即可。
(1)__enter__
方法在进入上下文时被调用,该函数的返回结果可被as
获取
(2)__exit__
方法在退出上下文时被调用。
(1)异常忽略 — exit实现
def __exit__(self, exc_type, exc_val, exc_tb):
"""
:param exc_type: 异常类型
:param exc_val: 异常对象
:param exc_tb: 错误的堆栈对象
:return: bool
"""
if exc_type==KeyError:
return True
return False
__exit__
方法的返回值为True则忽略异常不再向上层汇报,为False时则正常抛出。
(2)异常忽略 — @contextmanager装饰器
将生成器函数直接转换为上下文管理器。
以yield
关键字为界:
(1)yield
前的逻辑会在进入管理器时执行,类似__enter__
(2)yield
后的逻辑会在退出管理器时执行,类似__exit__
(3)处理异常必须在try
语句包裹yield
语句