使用 Python 实现简单的 switch/case 语句的方法


Posted in Python onSeptember 17, 2018

在Python中是没有Switch / Case语句的,很多人认为这种语句不够优雅灵活,在Python中用字典来处理多条件匹配问题字典会更简单高效,对于有一定经验的Python玩家不得不承认,的确如此。

但今天我们还是来看看如果一定要用Python来Switch / Case,可以怎么玩。

语法约束

我们先定义一下Switch/Case应该怎么表达,为了简单我们可以让它长成这样。

def cn():
  print('cn')
def us():
  print('us')
switch(lang).case('cn',cn)
truetruetrue.case('us',us)
   .default(us)

类实现一

通过以上约束,我们可以把switch当成一个类来实现,传入的参数在构造函数里处理,然后再分别实现case和default方法即可。

class switch(object):
  def __init__(self, case_path):
    self.switch_to = case_path
    self._invoked = False
  def case(self, key, method):
    if self.switch_to == key and not self._invoked:
      self._invoked = True
      method()
    return self
  def default(self, method):
    if not self._invoked:
      self._invoked = True
      method()

在构造函数中我们记住了 case_path 和执行状态 _invoked ,在 case() 里如果当前的 key 和 switch_to 匹配并且函数没有被执行过,那么就更新 _invoked 并执行对应的方法。在 default() 里检查一下 _invoked ,如果从没执行过,那么就调用 default 分支的函数。

看上去还不错,我们来试用一下。

switch('cn').case('cn',cn).case('us',us).default(fail)
>>> cn
switch('us').case('cn',cn).case('us',us).default(fail)
>>> cn
switch('jp').case('cn',cn).case('us',us).default(fail)
>>> fail
switch('cn').case('cn',cn).case('us',us)
>>> cn

让我们来看几个奇葩一点的case。

# duplicate case
switch('us').case('us',cn).case('us',us).default(fail)
>>> cn
def cn() return 'cn'
def us() return 'us'
# return value
result = switch('cn').case('cn',cn).case('us',us)
result
>>> <python_switch_case.switch object at 0x11034fb70>

发现了没有,上面的实现不会处理重复的case,当然你可以加强一下case方法,最好是抛出异常,其他编程语言通常都这样做。

第二个问题,你希望从case里拿到返回值,像上面的写法是没希望了,因为扔掉了。我们可以考虑在switch类里加一个result的变量来保存执行结果。

class switch(object):
  def __init__(self, case_path):
    ...
    self.result = None
  def case(self, key, method):
    ...
    self.result = method()
  ...

在调用结束后,就可以通过 result 拿到结果了。

_ = switch('cn').case('cn',cn).case('us',us)
_.result
>>> cn

类实现二

我大概在网上搜了一下,你还可以参考 Brian Beck 通过类来实现Swich/Case。

class switch(object):
  def __init__(self, value):
    self.value = value
    self.fall = False
  def __iter__(self):
    """Return the match method once, then stop"""
    yield self.match
    raise StopIteration
  def match(self, *args):
    """Indicate whether or not to enter a case suite"""
    if self.fall or not args:
      return True
    elif self.value in args:
      self.fall = True
      return True
    else:
      return False
c = 'z'
for case in switch(c):
  if case('a'): pass # only necessary if the rest of the suite is empty
  if case('c'): pass
  # ...
  if case('y'): pass
  if case('z'):
    print("c is lowercase!")
    break
  if case('A'): pass
  # ...
  if case('Z'):
    print("c is uppercase!")
    break
  if case(): # default
    print("I dunno what c was!")

这种实现相对复杂一点,而且用起来也不是很舒服,又需要for又需要if(还不如直接if/else痛快)。当然也有好处,就是可以把相同结果的case放一起,而且case里可以写更多东西,不仅仅是一个方法名。

写在最后

最后我们还是回到Python推崇的方法来处理switch/case问题,一般我们可以通过字典来处理这种多分支的问题,举例说明。

MAPPING = {
  'cn': cn,
  'us': us
}
lang = 'cn'
result = MAPPING.get(lang, default=us)

是不是一目了然,不仅易于阅读也易于维护。在字典中key是唯一的,value可以是任意类型的数据,可以是类或者是方法,所以足够灵活。

下面通过代码再次学习python语言switch-case

 初学python语言,竟然很久才发现python没有switch-case语句,查看官方文档说是可以用if-elseif-elseif。。。。代替。

讲真,这都不是问题。不就是一个条件判断吗。用if-elseif-.......肯定没问题,同时也用其他的解决方案,比较简单的就是利用

字典来实现同样的功能。写一个字典,每个key对应的值是一个方法。如switch =

{"valueA":functionA,"valueB":functionB,"valueC":functionC}

调用时可以像这样

try:
switch["value"]() #执行相应的方法。
except KeyError as e:

pass 或 functionX #执行default部分

简单代码如下:

switch = {
   "a":lambda x:x*2,
   "b":lambda x:x*3,
   "c":lambda x:x**x
 } 
 try:
   swtich["c"](6)
 except KeyError as e:
   pass

如果不嫌麻烦自己写一个swtich类来实现也没问题......不过真有这个必要吗

总结

以上所述是小编给大家介绍的使用 Python 实现简单的 switch/case 语句的方法,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

Python 相关文章推荐
Python中lambda的用法及其与def的区别解析
Jul 28 Python
python复制文件的方法实例详解
May 22 Python
python制作小说爬虫实录
Aug 14 Python
python logging日志模块的详解
Oct 29 Python
对numpy.append()里的axis的用法详解
Jun 28 Python
Python中logging.NullHandler 的使用教程
Nov 29 Python
Python当中的array数组对象实例详解
Jun 12 Python
python开头的coding设置方法
Aug 08 Python
python爬虫 爬取超清壁纸代码实例
Aug 16 Python
复化梯形求积分实例——用Python进行数值计算
Nov 20 Python
python中Lambda表达式详解
Nov 20 Python
numpy:找到指定元素的索引示例
Nov 26 Python
深入理解Python中的 __new__ 和 __init__及区别介绍
Sep 17 #Python
python实现简单http服务器功能
Sep 17 #Python
python实现自动登录
Sep 17 #Python
python发送告警邮件脚本
Sep 17 #Python
python实现zabbix发送短信脚本
Sep 17 #Python
python通过zabbix api获取主机
Sep 17 #Python
Python从ZabbixAPI获取信息及实现Zabbix-API 监控的方法
Sep 17 #Python
You might like
PHP下编码转换函数mb_convert_encoding与iconv的使用说明
2009/12/16 PHP
推荐一款MAC OS X 下php集成开发环境mamp
2014/11/08 PHP
php一维二维数组键排序方法实例总结
2014/11/13 PHP
ThinkPHP公共配置文件与各自项目中配置文件组合的方法
2014/11/24 PHP
PHP数据库连接mysql与mysqli对比分析
2016/01/04 PHP
php处理单文件、多文件上传代码分享
2016/08/24 PHP
PHP扩展安装方法步骤解析
2020/11/24 PHP
javascript据option的value值快速设定初始的selected选项
2007/08/13 Javascript
eval与window.eval的差别分析
2011/03/17 Javascript
使用javascript获取flash加载的百分比的实现代码
2011/05/25 Javascript
基于jQuery的前端数据通用验证库
2011/08/08 Javascript
JQuery中如何传递参数如click(),change()等具体实现
2013/04/28 Javascript
JS教程:window.location使用方法的区别介绍
2013/10/04 Javascript
jQuery的:parent选择器定义和用法
2014/07/01 Javascript
JavaScript数组前面插入元素的方法
2015/04/06 Javascript
jquery插件unobtrusive实现片段式加载
2015/06/15 Javascript
JS去除空格和换行的正则表达式(推荐)
2016/06/14 Javascript
如何判断出一个js对象是否一个dom对象
2016/11/24 Javascript
详解Nodejs 通过 fs.createWriteStream 保存文件
2017/10/10 NodeJs
vue注册组件的几种方式总结
2018/03/08 Javascript
vuejs 动态添加input框的实例讲解
2018/08/24 Javascript
浅谈webpack4 图片处理汇总
2018/09/12 Javascript
jQuery模仿ToDoList实现简单的待办事项列表
2019/12/30 jQuery
微信小程序实现录制、试听、上传音频功能(带波形图)
2020/02/27 Javascript
[20:21]《一刀刀一天》第十六期:TI国际邀请赛正式打响,总奖金超过550万
2014/05/23 DOTA
python linecache 处理固定格式文本数据的方法
2019/01/08 Python
查看python安装路径及pip安装的包列表及路径
2019/04/03 Python
python实现祝福弹窗效果
2019/04/07 Python
Django中使用haystack+whoosh实现搜索功能
2019/10/08 Python
TensorFlow2.X结合OpenCV 实现手势识别功能
2020/04/08 Python
Python新手学习raise用法
2020/06/03 Python
浅谈matplotlib 绘制梯度下降求解过程
2020/07/12 Python
美国花园雕像和家居装饰网上商店:Design Toscano
2019/03/09 全球购物
经理助理岗位职责
2014/03/05 职场文书
幼儿园个人总结
2015/02/28 职场文书
幼儿园中班教育随笔
2015/08/14 职场文书