三元表达式
在很多情况下,使用普通的 if/else
语句的代码可读性确实更好。盲目追求三元表达式很容易诱惑你写出复杂、可读性差的代码。
所以,请记得只用三元表达式处理简单的逻辑分支。比如如下代码是适合使用三元表达式的:
language = "python" if you.favor("dynamic") else "golang"
对于绝大多数情况,还是使用普通的 if/else
语句吧。比如下面代码
self.enabled = True if kwargs['enable'] and kwargs['already_started'] == 'Yes' and self.checked == 1 else False
self.need_use_ssl = True if kwargs['use_ssl'] is True else False if kwargs['allow_insecure'] is False else True
上面两种情况使用三元表达式反而将代码写的很长,第二行还有条件嵌套,使代码更加不容易理解
下面章节会介绍简化 if-else
statement的具体技巧,这里我们记住三元表达式需要慎用。
重构 if…else 语句
Nested if-else hell
如果你对这种代码非常欣赏,并且内心有一种想要搞清楚它在干什么的冲动,请略过这章的内容。
使用卫语语句简化函数
def get_payment_amount(account: Account) -> float:
result = 0.0
if account.is_deactivated:
result = get_deacticated_amount(account)
else:
if account.is_separated:
result = get_separated_amount(account)
else:
if account.is_retired:
result = get_retired_amount(account)
else:
result = get_normal_amount(account)
return result
该代码段存在大量的条件嵌套,使用卫语简化这个代码段,也叫提前结束(early ending)
def get_payment_amount(account: Account) -> float:
if account.is_deactivated:
return get_deacticated_amount(account)
if account.is_separated:
return get_separated_amount(account)
if account.is_retired:
return get_retired_amount(account)
return get_normal_amount(account)
重构后,基本移除了所有的 else 语句,使整个函数变得更加清晰,容易理解。
使用多态替代条件表达式
以下代码段是判断一个员工是否能够休假的代码
def is_eligible_for_pto(employee: Employee) -> bool:
if employee.type == "Manager":
return employee.hired_days > 100 and employee.performance >= 0.8
elif employee.type == "Developer":
return employee.hired_days > 30 and employee.performance >= 0.6 and employee.is_on_duty is False
elif employee.type == "Intern":
return employee.hired_days > 7 and employee.performance >= 0.4 and employee.is_on_duty is False
elif employee.type == "Director":
return employee.is_on_duty is False
return False
我们已经用卫语替代条件分支的方法重构过了,但是这个代码依然看起来很繁琐,因为每个条件分支都有很长很复杂的判断逻辑,而对于这个函数来说,它需要的知识太多了,必须知道employee 这个对象的属性,以及影响 PTO 的相关条件才能判断出来。
我们如果使用多态的方式来重构这段代码,能够将判断逻辑封装在具体的子类中,并且只需要在函数中调用相应接口即可,并不需要知道关于 Employee这个对象更多的信息即可完成工作。
class Employee:
type: str
hired_days: int
performance: float
@property
def pto_condition(self) -> bool:
return False
class Manager(Employee):
@property
def pto_condition(self) -> bool:
return self.hired_days > 100 and \
self.performance >= 0.8
class Developer(Employee):
@property
def pto_condition(self) -> bool:
return self.hired_days > 30 and \
self.performance >= 0.6 and \
self.is_on_duty is False
class Intern(Employee):
@property
def pto_condition(self) -> bool:
return self.hired_days > 7 and \
self.performance >= 0.4 and \
self.is_on_duty is False
class Director(Employee):
@property
def pto_condition(self) -> bool:
return self.is_on_duty is False
def is_eligible_for_pto(employee: Employee) -> bool:
return employee.pto_condition is True
用否定条件判断代替 else 分支
bad:
def _to_list(src):
if src:
return src if isinstance(src, list) else [src]
else:
return []
good:
def _to_list(src):
if not src:
return []
return src if isinstance(src, list) else [src]
逻辑重构
使用 for...else
替代 flag 以简化代码
bad:
flag = False
for index in range(10):
if index == 20:
flag = True
break
if not flag:
print("not found it")
good:
for index in range(10):
if index = 20:
break
else:
print("not found it")
使用字典替代条件判断
Python 中是不存在 switch 语句的,(新版本才支持),所以,当存在较多条件分支时,会写非常长的if
判断语句,比如像下面这种:
def global_events_dispatcher(event):
if event == 'event_1':
return call_event_handler_1()
elif event == 'event_2':
return call_event_handler_2()
elif event == 'event_3':
return call_event_handler_3()
else:
raise UnknownEventError
这种代码不但有多个条件分支,看起来比较复杂,同时也是重复代码的一种表现。
我们可以利用 Python 的dict
对这段条件分支代码进行重构,减少条件分支,去除重复代码。
def global_events_dispatcher(event):
event_handlers = {
'event_1': call_event_handler_1,
'event_2': call_event_handler_2,
'event_3': call_event_handler_3,
}
if event_handlers.get(event, None):
func = event_handlers(event)
return func()
raise UnknownEventError
重构后代码只有一个条件判断语句,函数的行为没有发生任何变化,让整个函数的逻辑变得更加容易理解。
补充
对于如何重构条件分支语句,可以延伸阅读《重构:改善既有代码的设计(第 2 版)》第 10 章:简化条件逻辑,里面罗列了多种常见的简化条件逻辑的重构手法,并配有大量的例子,推荐阅读。
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。