与我之前使用的所有语言都不同,Python没有switch/case语句。为了达到这种分支语句的效果,一般方法是使用字典映射:
def numbers_to_strings(argument):
switcher = {
0: "zero",
1: "one",
2: "two",
}
return switcher.get(argument, "nothing") 这段代码的作用相当于: function(argument){
switch(argument) {
case 0:
return "zero";
case 1:
return "one";
case 2:
return "two";
default:
return "nothing";
};
};
看起来,Python代码的对分支情况的处理方式比switch语句要更加简洁,但是我也可以认为它更晦涩难懂。我刚开始写Python代码的时候,总觉得怪怪的。后来时间长了,使用字典key作为分支条件,就越来越得心应手,越来越习惯了。
字典映射到函数
在Python中,字典可以映射到函数或则lambda表达式
def zero():
return "zero"
def one():
return "one"
def numbers_to_functions_to_strings(argument):
switcher = {
0: zero,
1: one,
2: lambda: "two",
}
# Get the function from switcher dictionary
func = switcher.get(argument, lambda: "nothing")
# Execute the function
return func()
虽然上例中zero()和one()中的代码非常简单,但是许多Python程序都用字典映射来分配处理复杂的程序流程。
指派到类方法
在一个类里,如果我们不知道该调用哪个方法,那么我们可以使用一个分派方法在运行时决定:
class Switcher(object):
def numbers_to_methods_to_strings(self, argument):
"""Dispatch method"""
# prefix the method_name with 'number_' because method names
# cannot begin with an integer.
method_name = 'number_' + str(argument)
# Get the method from 'self'. Default to a lambda.
method = getattr(self, method_name, lambda: "nothing")
# Call the method as we return it
return method()
def number_0(self):
return "zero" def number_1(self):
return "one" def number_2(self):
return "two"
漂亮的实现,对吧?
官方解释
官方说法是,“你可以用一系列的 if...elif...elif...else 语句来处理这些问题”。而且还能使用字典映射到函数,分派到类方法。
令人疑惑的是,官方只给出了替代方案,而并没有解释为什么。换句话说,就是“无可奉告“。在我看来,官方想要表达的意思其实就是”Python不需要case语句“。
真相是什么?
然而我听得最多的说法是,switch/case语句非常难以调试。
但是稍微思考一下就知道这种说法是站不住脚的。只需要假想一下,你使用了一个重重嵌套的巨大字典用来处理分支逻辑;如果这个字典元素超过了100个,那么它的调试难度其实并不低于100个case语句。
或许因为字典映射速度更快?
然并卵,Python没有switch/case语句,没法测试,跳过这一点。
Python这种做法的巨大优势
经验之谈,我经常会碰到一些情景,Python的做法比switch/case语句要优雅有效的多,那就是在我需要在运行时增删映射项的时候。碰到需要这么做的时候,我的Python技能就碉堡了,可以动态的改变字典映射和类方法分派调用。有了这些心得之后,我在也没有怀念过switch/case语句。
最终章
对我而言,使用Python的经验迫使我使用字典映射,而我亦从中因祸得福。没有switch/case语句的苦恼使得我产生了以前没有过的想法、实现了以前没开发过的功能。
总而言之,Python switch/case语句的缺失,使我成为了更好的程序员;而这种开发生态,就是我所期望的比“官方解释”更好的答案
未来CTO关注我CTO之路从此开始微信号:wlaicto