本篇文章起源于StackOverflow上一个热度很是高的问题:python
@Ray Vega (提问者)编程语言
举例说明,如今我获得了以下代码:性能
a = []我如何该检查
a
是否为空?测试
面对这个问题,各路高手给出了不尽相同的回答。编码
最高票答案十分简洁:翻译
@Patrick (答题者)设计
if not a: print("List is empty")利用空列表的隐式布尔值是一个很是Pythonic的方式。code
排名第二的答案与第一观点相同,并以PEP 8做为依据,说明不只是列表,Python中的内置序列类型都有推荐的作法:开发
@Harley Holcombe (答题者)
PEP 8 风格指南 给出了推荐的Pythonic的方法(其中Yes 表示推荐, No表示不推荐):
对序列数据类型(字符串,列表,元组),利用空列表隐式为
False
的事实Yes: if not seq: if seq: No: if len(seq): if not len(seq):
然而,排名第三的答案给出了不一样的见解:
我更推荐显式的方法:
if len(li) == 0: print('the list is empty')这种方式明确声明了
li
是一个序列类型的变量,而且咱们是在检查它的长度。而if not li
的问题在于,它会给我li
是一个布尔类型变量的印象。
那么,判断列表(序列)是否为空的正确姿式究竟是什么呢?这貌似只是一个编码风格的问题,但咱们分别从两类不一样见解的出发点挖掘更深层次的缘由,可让本身更明确地选择适合本身的风格。
PEP,全称Python Enhancement Proposals (翻译过来就是Python加强建议书),有兴趣的读者能够直接阅览PEP原文。 PEP本质上是一份Python的官方文档库,给Python社区提供信息,或者描述Python的新特性或开发进展。而PEP 8是这个文档库中的一员,专门用于描述Python的编码规范,这里规范是指官方推荐的,被认为是更符合Python设计哲学的各类实践。
一样实现相同的功能,不一样编程语言的倾向于使用不一样的风格,这是由于每种语言都有自身的设计目的,而Python的设计目的很是明显:优雅,简单,可读。正如PEP 20(另外一份PEP)Python之禅中所说:
简单优于复杂
因而,依据序列长度是否为0将序其隐式转化为布尔值,成为Python实现中的特性之一,并成为官方推荐的判断序列是否为空的Pythonic方式。
关于Python是如何作到序列类型乃至全部类型到布尔值的隐式转化的,我会专门就此问题写文讨论,欢迎关注。
那么为何还会有人提出明确使用看上去复杂的if len(li) == 0
来判断,而且还有不少人表示赞同呢?这其实来源于Python语言的动态类型特性。
关于什么是动态类型,我也会另外专门讨论,在这里,咱们只需阐明,动态类型带来了一个弊端,咱们没法对变量在程序中某一位置的类型进行准确判断。在阅读Python代码的过程当中,咱们可能最为头痛的问题就是:这里这个变量是什么(类型)???惟一留给咱们的线索也许只有变量名了。而在静态类型语言,如Java中,一个变量的类型从其声明时是肯定的,在程序中不会发生改变。
回到咱们的问题,if not li
,看到这段代码的程序猿可能会疑惑,这里的li
变量是什么,是一个布尔类型?仍是一个整型?这里的测试是在干什么?而if len(li) == 0
能够很大程度上进行提示:这大抵是个容器性质的变量,咱们在作的大抵是判断其元素数量是否为0.
讨论到这里,咱们仍然只能说,Python中如何判断一个列表是否为空,是一个与风格和习惯有关的问题,可是深刻探究咱们发现,风格和习惯不是目的,而是手段,代码最终是服务于编码者和阅读者的,抛开性能问题,只从可读性出发,你但愿阅读这份代码的人接受到的是什么,是简单优雅,仍是信息提示,这才是比所谓Pythonic更值得思考的问题。
获取最新文章更新,欢迎关注个人公众号: StackOverflow Daily