Python五个隐藏的特性,你可能从未听说过

 前言

在本文中,我将向您展示Python中很常见的5个特性。有经验的Python开发人员可能认识其中一些。然而,这对其他人仍将是未知的。

Python五个隐藏的特性,你可能从未听说过

1...

是的,你没看错,在Python中...是一个有效的构造。...是称为省略号的单例对象。如果你把它输入到Python解释器中,你可以看到它:

>>> ... 
Ellipsis 

根据官方文档,省略号是“一种特殊值,主要与用户定义容器数据类型的扩展切片语法结合使用”。它有两个主要的用例。一种是在空函数中充当占位符体。另一个是Numpy,作为一个切片项,就像文档中描述的那样。

函数的占位符

def my_awesome_function(): 
    ... 

这相当于:

def my_awesome_function(): 
    Ellipsis 

还有这个:

def my_awesome_function(): 
    pass 

注意,我不是说pass =…我只是说作为函数体,结果是一样的。事实上,您可以使用任何东西作为占位符。

Numpy

下面的代码基本上意味着创建一个矩阵数组。每个矩阵是3×3。然后获取所有最内部矩阵的第二列(numpy数组基于0)。

import numpy as np 
>>> array = np.arange(27).reshape(3, 3, 3) 
>>> array 
array([[[ 0,  1,  2], 
        [ 3,  4,  5], 
        [ 6,  7,  8]], 
 
       [[ 9, 10, 11], 
        [12, 13, 14], 
        [15, 16, 17]], 
 
       [[18, 19, 20], 
        [21, 22, 23], 
        [24, 25, 26]]]) 
>>> array[..., 1]  
array([[ 1,  4,  7], 
       [10, 13, 16], 
       [19, 22, 25]]) 
>>> # This is equivalent to 
>>> array[:, :, 1]  
array([[ 1,  4,  7], 
       [10, 13, 16], 
       [19, 22, 25]]) 

2一个优雅的解包

可迭代解包是一种非常方便的特性,已经存在一段时间了。大多数人使用它来解包包含多个项的可迭代对象。例如,考虑以下用例。

>>> a, *b, c = range(1, 11) 
>>> a 
1 
>>> c 
10 
>>> b 
[2, 3, 4, 5, 6, 7, 8, 9] 

或者是:

>>> a, b, c = range(3) 
>>> a 
0 
>>> b 
1 
>>> c 
2 

但有一个很好的用例,很多人都没有利用它,那就是拆封单个迭代器。为什么这很有用?恕我直言,它使代码更优雅了一些。

而不是这样做:

>>> lst = [1] 
>>> a = lst[0] 
>>> a 
1 
>>> (a, ) = lst 
>>> a 
1 

你可以这样做:

>>> lst = [1] 
>>> [a] = lst 
>>> a 
1 

我知道这可能看起来很傻,但至少对我来说,它看起来更优雅。

3你能让这个列表躺平吗?

扁平化列表有几种方法。最简单的是使用列表理解。

>>> l = [[1, 2, 3], [4, 5, 6], [7, 8, 9]] 
>>> flattened = [elem for sublist in l for elem in sublist] 
>>> flattened 
[1, 2, 3, 4, 5, 6, 7, 8, 9] 

如果您更倾向于函数式编程,您可以使用减速器。

>>> from functools import reduce 
>>> reduce(lambda x,y: x+y,l) 
[1, 2, 3, 4, 5, 6, 7, 8, 9] 

然而,还有另一种方法。你可以用sum函数!

>>> sum(l, []) 
[1, 2, 3, 4, 5, 6, 7, 8, 9] 

这是因为sum函数遍历列表中的每个元素,并将它们与作为第二个参数传递的默认值连接起来。因为Python中的列表可以用+操作符连接,所以你得到的结果是这样的:

>>> sum(l, []) ==> [] + [1, 2, 3] + [4, 5, 6] + [7, 8, 9] 
[1, 2, 3, 4, 5, 6, 7, 8, 9] 

尽管这个技巧很高明,但它绝不是可读的。而且,它的性能也很糟糕。

4else

else语句可以用于几个目的。很少有人知道,但是你可以在经典的“if else”块之外使用它。Python允许它用于循环和异常块。

循环

Python有两个不同的循环,for和while。两者都可能是“坏的”。也就是说,如果满足了某个条件,就可以跳出循环。例如:

In [7]: while a < 10: 
   ...:     if a == 3: 
   ...:         print("a == 3. exiting loop.") 
   ...:         break 
   ...:     a += 1 
   ...:  
a == 3. exiting loop. 

现在,假设我们要找一个特定的条件。如果满足该条件,则将结果保存在一个名为found的标志中。然后,如果我们没有找到它,我们打印一条消息。

found = False 
a = 0 
 
while a < 10: 
    if a == 12: 
        found = True 
    a += 1 
if not found: 
    print("a was never found") 

因为a永远不会变成12,所以程序输出a永远不会找到。

好,但是我们在这里怎么用else呢?

else可以用来替换标志。基本上,我们实际需要的是运行循环,如果没有找到,则打印一条消息。

a = 0 
 
while a < 10: 
    if a == 12: 
        break 
    a += 1 
else: 
    print("a was never found") 

由于它适用于任何循环,所以您可以使用for而不是while。

for a in range(10): 
    if a == 12: 
        break 
    a += 1 
else: 
    print("a was never found") 

异常

Python中的else是如此通用,你甚至可以使用try…except。这里的思想是捕获异常不发生的情况。

In [13]: try: 
    ...:     {}['lala'] 
    ...: except KeyError: 
    ...:     print("Key is missing") 
    ...: else: 
    ...:     print("Else here") 
    ...:  
Key is missing 

在这个例子中,我们尝试在一个空字典中查找名为“lala”的键。由于“lala”不存在,代码将引发一个KeyError异常。当我在IPython中运行这段代码时,得到了预期的结果。

如果程序没有引发异常呢?

In [14]: try: 
    ...:     {'lala': 'bla'}['lala'] 
    ...: except KeyError: 
    ...:     print("Key is missing") 
    ...: else: 
    ...:     print("Else here") 
    ...:  
Else here 

现在我们可以看到它的实际应用。{' lala ': ' bla '}[' lala ']块不会引发KeyError,所以else就起作用了。

5比较

这是我最喜欢的一个,老实说,没有那么隐蔽。与许多编程语言(如Java、C或c++)不同,Python允许链式比较运算符。假设你有一个变量x,它的值是10。现在,假设你想断言x在一个范围内,比如5..20。你可以这样做:

In [16]: x = 10 
In [17]: if x >= 5 and x <= 20: 
    ...:     print("x is within range") 
    ...: else: 
    ...:     print("x is outside range") 
    ...:  
is within range 

事实证明,这可以通过将运算符链接起来来简化。所以,我们可以重构代码为:

In [18]: if 5 <= x <= 20: 
    ...:     print("is within range") 
    ...: else: 
    ...:     print("x is outside range") 
    ...:  
is within range 

相关推荐