2.15. 更多知识

未匹配的标注

更多知识

到目前为止,我们已经涵盖了 Python 中你将会使用的各个方面的大部分内容。 在本章中,我们将介绍一些将使我们的 Python 知识更全面的内容。

传递元组

你曾经希望能从函数中返回两个不同的值吗? 这当然可以。 你所要做的就是使用一个元组。

>>> def get_error_details():
...     return (2, 'details')
...
>>> errnum, errstr = get_error_details()
>>> errnum
2
>>> errstr
'details'

请注意, a, b = <某些表达式> 会将表达式的结果解析为两个值组成的元组。

这也意味着在 Python 中交换两个变量的最快方法是:

>>> a = 5; b = 8
>>> a, b
(5, 8)
>>> a, b = b, a
>>> a, b
(8, 5)

魔术方法

有一些方法,如 __init____del__ 方法,它们在类中具有特殊意义。

魔术方法用于模仿内置类型的某些行为。 例如,如果你想为你的类使用 x[key] 索引操作(就像你列表和元组使用它一样),那么你所要做的就是实现 __getitem __() 方法,这样你的工作就已经完成了。 如果你思考一下,这就是 Python 本身为 list 类所做的事情!

下表列出了一些有用的魔术方法。 如果你想了解所有的魔术方法, 参考手册

  • __init__(self, ...)

    • 在返回新创建可以使用的对象之前调用此方法。
  • __del__(self)

    • 在对象被销毁之前调用(具有不可预测时机,所以避免使用它)
  • __str__(self)

    • 当我们使用 print 函数或使用 str() 时调用。
  • __lt__(self, other)

    • 使用小于( less than )运算符(<)时调用。 同样,所有运算符都有特殊的方法(+,>等)
  • __getitem__(self, key)

    • 使用 x[key] 索引操作时调用。
  • __len__(self)

    • 当内置的 len() 函数用于序列对象时调用。

单个语句块

我们已经看到,每个语句块都通过其自己的缩进级别与其余语句区分开来。 但有一点需要注意,如果你的语句块只包含一个语句,那么您可以在条件语句或循环语句的同一行上指定它。 以下示例应该说明这一点:

>>> flag = True
>>> if flag: print('Yes')
...
Yes

请注意,单个语句是就地使用的,而不是作为单独的块使用。 虽然,你可以使用它来使你的程序更小,但我强烈建议避免使用这种快捷方法,除了错误检查。主要是因为如果使用适当的缩进,添加额外的语句将更容易。

Lambda 格式

lambda 语句用于创建新的函数对象。 基本上, lambda 采用一个参数,后跟一个表达式。 Lambda 成为函数的函数体。 新函数返回此表达式的值。

示例 (保存为 more_lambda.py ):

points = [{'x': 2, 'y': 3},
          {'x': 4, 'y': 1}]
points.sort(key=lambda i: i['y'])
print(points)

输出:

$ python more_lambda.py
[{'y': 1, 'x': 4}, {'y': 3, 'x': 2}]

工作原理

请注意, listsort 方法可以采用 key 参数来确定列表的排序方式(通常我们只知道升序或降序)。 在我们的例子中,我们想要进行自定义排序,为此我们需要编写一个函数。 我们使用 lambda 表达式创建一个新函数,而不是为仅在这一个地方使用的函数编写单独的 def 块。

列表推导

列表推导用于从现有列表中导出新列表。 假设你有一个数字列表,并且你希望获得一个相应的新列表,其中所有数字仅在数字本身大于2时乘以2。列表推导是这种情况的理想选择。

示例(保存为 more_list_comprehension.py ):

listone = [2, 3, 4]
listtwo = [2*i for i in listone if i > 2]
print(listtwo)

输出:

$ python more_list_comprehension.py
[6, 8]

工作原理

这里,我们通过在满足某些条件时指定要完成的操作( 2*i )来推导出一个新的列表( if i > 2 )。 请注意,原始列表保持不变。

使用列表推导的优点是,当我们循环处理列表的每个元素并将其存储在新列表中时,它减少了所需样板代码的量。

在函数中接收元组和字典

有一种特殊的方法可以分别使用 *** 前缀将参数作为元组或字典接收到函数中。 当在函数中使用可变数量的参数时,这很有用。

>>> def powersum(power, *args):
...     '''返回每个参数指定幂次的总和。'''
...     total = 0
...     for i in args:
...         total += pow(i, power)
...     return total因为我们在`args`变量上有一个`*`前缀,所有传递给函数的额外参数都作为元组存储在`args`中。 如果使用了`**`前缀,那么额外的参数将被认为是字典的键/值对。
...
>>> powersum(2, 3, 4)
25
>>> powersum(2, 10)
100

因为我们在 args 变量上有一个 * 前缀,所有传递给函数的额外参数都作为元组存储在 args 中。 如果使用了 ** 前缀,那么额外的参数将被作为字典的键/值对。

assert 语句

assert语句用于断言某值为 True 。 例如,如果您非常确定您正在使用的列表中至少有一个元素并且想要检查它,并且如果不是 True 则引发错误,那么 assert 语句在这种情况下是理想的。 当 assert 语句失败时,会引发 AssertionErrorpop() 方法删除并返回列表中的最后一项。

>>> mylist = ['item']
>>> assert len(mylist) >= 1
>>> mylist.pop()
'item'
>>> assert len(mylist) >= 1
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AssertionError

assert 语句应该是明智地使用。 大多数情况下,最好能捕获异常、处理问题或向用户显示错误消息然后退出。

装饰器

装饰器是用于包装函数的快捷方式。 这有助于一遍又一遍地使用相同的代码『包装』功能。 例如,我为自己创建了一个 retry 装饰器,我能应用于任何函数,如果在运行期间抛出任何异常,则会再次重试,直到最多 5 次并且每次重试之间有间隔。 这对于你尝试向远程计算机进行网络连接的情况特别有用:

from time import sleep
from functools import wraps
import logging
logging.basicConfig()
log = logging.getLogger("retry")

def retry(f):
    @wraps(f)
    def wrapper_function(*args, **kwargs):
        MAX_ATTEMPTS = 5
        for attempt in range(1, MAX_ATTEMPTS + 1):
            try:
                return f(*args, **kwargs)
            except:
                log.exception("Attempt %s/%s failed : %s",
                              attempt,
                              MAX_ATTEMPTS,
                              (args, kwargs))
                sleep(10 * attempt)
        log.critical("All %s attempts failed : %s",
                     MAX_ATTEMPTS,
                     (args, kwargs))
    return wrapper_function

counter = 0

@retry
def save_to_database(arg):
    print("Write to a database or make a network call or etc.")
    print("This will be automatically retried if exception is thrown.")
    global counter
    counter += 1
    # 这将在第一次调用中抛出异常
    # 并且在第二次调用中工作正常(即重试)
    if counter < 2:
        raise ValueError(arg)

if __name__ == '__main__':
    save_to_database("Some bad value")

输出:

$ python more_decorator.py
Write to a database or make a network call or etc.
This will be automatically retried if exception is thrown.
ERROR:retry:Attempt 1/5 failed : (('Some bad value',), {})
Traceback (most recent call last):
  File "more_decorator.py", line 14, in wrapper_function
    return f(*args, **kwargs)
  File "more_decorator.py", line 39, in save_to_database
    raise ValueError(arg)
ValueError: Some bad value
Write to a database or make a network call or etc.
This will be automatically retried if exception is thrown.

工作原理

参见:

Python 2 与 Python 3 的不同

参见:

小结

我们在本章中介绍了 Python 一些更多的功能,但我们还没有涵盖 Python 的所有功能。 但是,在这个阶段,我们已经涵盖了你将在实践中使用的大部分内容。 这足以让你开始使用你要创建的任何程序。

接下来,我们将讨论如何进一步探索 Python 。

本文章首发在 LearnKu.com 网站上。

本译文仅用于学习和交流目的,转载请务必注明文章译者、出处、和本文链接
我们的翻译工作遵照 CC 协议,如果我们的工作有侵犯到您的权益,请及时联系我们。

原文地址:https://learnku.com/docs/byte-of-python/...

译文地址:https://learnku.com/docs/byte-of-python/...

上一篇 下一篇
讨论数量: 0
发起讨论 只看当前版本


暂无话题~