翻译:Python 开发者 - 一汀, 英文:Trey Hunner

http://blog.jobbole.com/114655/

Python开发者

在 Python 中有很多地方可以看到***。在某些情形下,无论是对于新手程序员,还是从其他很多没有完全相同操作符的编程语言迁移过来的人来说,这两个操作符都可能有点神秘。因此,我想讨论一下这些操作符的本质及其使用方式。

多年以来,***操作符的功能不断增强。在本文中,我将讨论目前这些操作符所有的使用方法,并指出哪些使用方法只能在目前的 Python 版本中应用。因此,如果你学习过 Python 2 中***的使用方法,那么我建议你至少浏览一下本文,因为 Python 3 中添加了许多***的新用途。

如果你是新接触 Python 不久,还不熟悉关键字参数(亦称为命名参数),我建议你首先阅读我有关Python中的关键字参数的文章。

不属于我们讨论范围的内容

在本文中, 当我讨论***时,我指的是*** 前缀 操作符,而不是 中缀 操作符。

也就是说,我讲述的不是乘法和指数运算:

>>> 2 * 5

10

>>> 2 ** 5

32

那么我们在讨论什么内容呢?

我们讨论的是***前缀运算符,即在变量前使用的***运算符。例如:

>>> numbers = [2, 1, 3, 4, 7]

>>> more_numbers = [*numbers, 11, 18]

>>> print(*more_numbers, sep=', ')

2, 1, 3, 4, 7, 11, 18

上述代码中展示了*的两种用法,没有展示**的用法。

这其中包括:

  1. 使用***向函数传递参数

  2. 使用***捕获被传递到函数中的参数

  3. 使用*接受只包含关键字的参数

  4. 使用*在元组解包时捕获项

  5. 使用*将迭代项解压到列表/元组中

  6. 使用**将字典解压到其他字典中

即使你认为自己已经熟悉* 和 **的所有使用方法,我还是建议你查看下面的每个代码块,以确保都是你熟悉的内容。在过去的几年里,Python 核心开发人员不断地为这些操作符添加新的功能,对于使用者来说很容易忽略* 和 **‘的一些新用法。

星号用于将可迭代对象拆分并分别作为函数参数

当调用函数时,*运算符可用于将一个迭代项解压缩到函数调用中的参数中:

>>> fruits = ['lemon', 'pear', 'watermelon', 'tomato']

>>> print(fruits[0], fruits[1], fruits[2], fruits[3])

lemon pear watermelon tomato

>>> print(*fruits)

lemon pear watermelon tomato

print(*fruits)代码行将fruits列表中的所有项作为独立的参数传递给print函数调用,甚至不需要我们知道列表中有多少个参数。

*运算符在这里远不止是语法糖而已。要想用一个特定的迭代器将所有项作为独立的参数传输,若不使用*是不可能做到的,除非列表的长度是固定的。

下面是另一个例子:

def transpose_list(list_of_lists):

return [

list(row)

for row in zip(*list_of_lists)

]

这里我们接受一个二维列表并返回一个“转置”的二维列表。

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

[[1, 2, 3], [4, 5, 6], [7, 8, 9]]

**操作符完成了类似的操作,只不过使用了关键字参数。**运算符允许我们获取键-值对字典,并在函数调用中将其解压为关键字参数。

>>> date_info = {'year': "2020", 'month': "01", 'day': "01"}

>>> filename = "{year}-{month}-{day}.txt".format(**date_info)

>>> filename '2020-01-01.txt' `

根据我的经验,使用**将关键字参数解压缩到函数调用中并不常见。我最常看到它的地方是在实现继承时:对uper()的调用通常包括***

如 Python 3.5 那样,在函数调用中,***都可以被多次使用。

有时,多次使用*会很方便:

>>> fruits = ['lemon', 'pear', 'watermelon', 'tomato']

>>> numbers = [2, 1, 3, 4, 7]

>>> print(*numbers, *fruits)

2 1 3 4 7 lemon pear watermelon tomato `

多次使用**也可以达到相似的效果:

>>> date_info = {'year': "2020", 'month': "01", 'day': "01"}

>>> track_info = {'artist': "Beethoven", 'title': 'Symphony No 5'}

>>> filename = "{year}-{month}-{day}-{artist}-{title}.txt".format(

...     **date_info,

...     **track_info,

... )

>>> filename

'2020-01-01-Beethoven-Symphony No 5.txt'

不过,在多次使用**时需要特别小心。Python 中的函数不能多次指定相同的关键字参数,因此在每个字典中与**一起使用的键必须能够相互区分,否则会引发异常。

星号用于压缩被传递到函数中的参数

在定义函数时,*运算符可用于捕获传递给函数的位置参数。位置参数的数量不受限制,捕获后被存储在一个元组中。

from random import randint

def roll(*dice):

return sum(randint(1, die) for die in dice)

这个函数接受的参数数量不受限制:

>>> roll(20)

18

>>> roll(6, 6)

9

>>> roll(6, 6, 6)

8

Python 的printzip函数接受的位置参数数量不受限制。*的这种参数压缩用法,允许我们创建像printzip一样的函数,接受任意数量的参数。

**运算符也有另外一个功能:我们在定义函数时,可以使用** 捕获传进函数的任何关键字参数到一个字典当中:

def tag(tag_name, **attributes):

attribute_list = [

f'{name}="{value}"'

for name, value in attributes.items()

]

return f"<{tag_name} {' '.join(attribute_list)}>"

** 将捕获我们传入这个函数中的任何关键字参数,并将其放入一个字典中,该字典将引用attributes参数。

>>> tag('a', href="http://treyhunner.com")

'<a href="http://treyhunner.com">'

>>> tag('img', height=20, width=40, src="face.jpg")

'<img height="20" width="40" src="face.jpg">'

只有关键字参数的位置参数

在 Python 3 中,我们现在拥有了一种特殊的语法来接受只有关键字的函数参数。只有关键字的参数是只能 使用关键字语法来指定的函数参数,也就意味着不能按照位置来指定它们。

在定义函数时,为了接受只有关键字的参数,我们可以将命名参数放在*后:

def get_multiple(*keys, dictionary, default=None):

return [

dictionary.get(key, default)

for key in keys

]

上面的函数可以像这样使用:

>>> fruits = {'lemon': 'yellow', 'orange': 'orange', 'tomato': 'red'}

>>> get_multiple('lemon', 'tomato', 'squash', dictionary=fruits,default='unknown')

['yellow', 'red', 'unknown']

参数dictionarydefault*keys后面,这意味着它们只能 被指定为关键字参数。如果我们试图按照位置来指定它们,我们会得到一个报错:

>>> fruits = {'lemon': 'yellow', 'orange': 'orange', 'tomato': 'red'}

>>> get_multiple('lemon', 'tomato', 'squash', fruits, 'unknown')

Traceback (most recent call last):

File "<stdin>", line 1, in <module>

TypeError: get_multiple() missing 1 required keyword-only argument: 'dictionary'

这种行为是通过 PEP 3102 被引入到 Python 中的。

没有位置参数关键字的参数

只使用关键字参数的特性很酷,但是如果您希望只使用关键字参数而不捕获无限的位置参数呢?

Python 使用一种有点奇怪的 单独* 语法来实现:

def with_previous(iterable, *, fillvalue=None):

"""Yield each iterable item along with the item before it."""

previous = fillvalue

for item in iterable:

yield previous, item

previous = item

这个函数接受一个迭代器参数,可以按照位置或名字来指定此参数(作为第一个参数),以及关键字参数fillvalue,这个填充值参数只使用关键字。这意味着我们可以像下面这样调用 with_previous:

>>> list(with_previous([2, 1, 3], fillvalue=0))

[(0, 2), (2, 1), (1, 3)]

但像这样就不可以:

>>> list(with_previous([2, 1, 3], 0))

Traceback (most recent call last):

File "<stdin>", line 1, in <module>

TypeError: with_previous() takes 1 positional argument but 2 were given `

这个函数接受两个参数,其中fillvalue参数必须被指定为关键字参数

我通常在获取任意数量的位置参数时只使用关键字参数,但我有时使用这个*强制按照位置指定一个参数。

实际上,Python 的内置sorted函数使用了这种方法。如果你查看sorted的帮助信息,将看到以下信息:

>>> help(sorted)

Help on built-in function sorted in module builtins:

sorted(iterable, /, *, key=None, reverse=False)

Return a new list containing all items from the iterable inascending order.

A custom key function can be supplied to customize the sort order, and the

reverse flag can be set to request the result in descending order.

sorted的官方说明中,有一个单独的*参数。

星号用于元组拆包

Python 3 还新添了一种 * 运算符的使用方式,它只与上面定义函数时和调用函数时*的使用方式相关。

现在,*操作符也可以用于元组拆包:

>>> fruits = ['lemon', 'pear', 'watermelon', 'tomato']

>>> first, second, *remaining = fruits

>>> remaining

['watermelon', 'tomato']

>>> first, *remaining = fruits

>>> remaining

['pear', 'watermelon', 'tomato']

>>> first, *middle, last = fruits

>>> middle

['pear', 'watermelon']

如果你想知道什么情况下可以在你自己的代码中使用它,请查看我关于 Python 中的 tuple 解包 文章中的示例。在那篇文章中,我将展示如何使用*操作符作为序列切片的替代方法。

通常当我教*的时候,我告诉大家只能在多重赋值语句中使用一个*表达式。实际来说这是不正确的,因为可以在嵌套解包中使用两个*(我在元组解包文章中讨论了嵌套解包):

>>> fruits = ['lemon', 'pear', 'watermelon', 'tomato']

>>> first, second, *remaining = fruits

>>> remaining

['watermelon', 'tomato']

>>> first, *remaining = fruits

>>> remaining

['pear', 'watermelon', 'tomato']

>>> first, *middle, last = fruits

>>> middle

['pear', 'watermelon']

但是,我从来没见过它有什么实际用处,即使你因为它看起来有点神秘而去寻找一个例子,我也并不推荐这种使用方式。

将此添加到 Python 3.0 中的 PEP 是 PEP 3132,其篇幅不是很长。

列表文字中的星号

Python 3.5 通过 PEP 448 引入了大量与*相关的新特性。其中最大的新特性之一是能够使用*将迭代器转储到新列表中。

假设你有一个函数,它以任一序列作为输入,返回一个列表,其中该序列和序列的倒序连接在了一起:

def palindromify(sequence):

return list(sequence) + list(reversed(sequence))

此函数需要多次将序列转换为列表,以便连接列表并返回结果。在 Python 3.5 中,我们可以这样编写函数:

def palindromify(sequence):

return [*sequence, *reversed(sequence)]

这段代码避免了一些不必要的列表调用,因此我们的代码更高效,可读性更好。

下面是另一个例子:

def rotate_first_item(sequence):

return [*sequence[1:], sequence[0]]

该函数返回一个新列表,其中给定列表(或其他序列)中的第一项被移动到了新列表的末尾。

* 运算符的这种使用是将不同类型的迭代器连接在一起的好方法。* 运算符适用于连接任何种类的迭代器,然而 + 运算符只适用于类型都相同的特定序列。

除了创建列表存储迭代器以外,我们还可以将迭代器转储到新的元组或集合中:

>>> fruits = ['lemon', 'pear', 'watermelon', 'tomato']

>>> (*fruits[1:], fruits[0])

('pear', 'watermelon', 'tomato', 'lemon')

>>> uppercase_fruits = (f.upper() for f in fruits)

>>> {*fruits, *uppercase_fruits}

{'lemon', 'watermelon', 'TOMATO', 'LEMON', 'PEAR','WATERMELON', 'tomato', 'pear'}

注意,上面的最后一行使用了一个列表和一个生成器,并将它们转储到一个新的集合中。在此之前,并没有一种简单的方法可以在一行代码中完成这项工作。曾经有一种方法可以做到这一点,可是并不容易被记住或发现:

两个星号用于字典文本

PEP 448 还通过允许将键/值对从一个字典转储到一个新字典扩展了**操作符的功能:

>>> date_info = {'year': "2020", 'month': "01", 'day': "01"}

>>> track_info = {'artist': "Beethoven", 'title': 'Symphony No 5'}

>>> all_info = {**date_info, **track_info}

>>> all_info

{'year': '2020', 'month': '01', 'day': '01', 'artist': 'Beethoven', 'title':'Symphony No 5'}

我还写了另一篇文章:在Python中合并字典的惯用方法。

不过,**操作符不仅仅可以用于合并两个字典。

例如,我们可以在复制一个字典的同时添加一个新值:

>>> date_info = {'year': '2020', 'month': '01', 'day': '7'}

>>> event_info = {**date_info, 'group': "Python Meetup"}

>>> event_info

{'year': '2020', 'month': '01', 'day': '7', 'group': 'Python Meetup'}

或者在复制/合并字典的同时重写特定的值:

>>> event_info = {'year': '2020', 'month': '01', 'day': '7', 'group':'Python Meetup'}

>>> new_info = {**event_info, 'day': "14"}

>>> new_info

{'year': '2020', 'month': '01', 'day': '14', 'group': 'Python Meetup'}

Python 的星号非常强大

Python 的 * 和 ** 运算符不仅仅是语法糖。 * 和 ** 运算符允许的某些操作可以通过其他方式实现,但是往往更麻烦和更耗费资源。而且 * 和 ** 运算符提供的某些特性没有替代方法实现:例如,函数在不使用 * 时就无法接受任意数量的位置参数。

在阅读了* 和 ** 运算符的所有特性之后,您可能想知道这些奇怪操作符的名称。不幸的是,它们的名字并不简练。我听说过* 被称为“打包”和“拆包“运算符。我还听说过其被称为“splat”(来自 Ruby 世界),也听说过被简单地称为“star”。

我倾向于称这些操作符为“星”和“双星”或“星星”。这种叫法并不能区分它们和它们的中缀关系(乘法和指数运算),但是通常我们可以从上下文清楚地知道是在讨论前缀运算符还是中缀运算符。

请勿在不理解* 和 ** 运算符的前提下记住它们的所有用法!这些操作符有很多用途,记住每种操作符的具体用法并不重要,重要的是了解你何时能够使用这些操作符。我建议使用这篇文章作为一个备忘单或者制作你自己的备忘单来帮助你在 Python 中使用解* 和 ** 。

喜欢我的教学风格吗?

想了解更多关于 Python 的知识吗?我每周通过实时聊天分享我最喜欢的 Python 资源、回答 Python 问题。

尚学堂推出《13天搞定Python网络爬虫》视频教程,学习成为Python爬虫工程师,薪资杠杠的!

Python中星号的本质和使用方式的更多相关文章

  1. python中逐行读取文件的最佳方式_Drupal_新浪博客

    python中逐行读取文件的最佳方式_Drupal_新浪博客 python中逐行读取文件的最佳方式    (2010-08-18 15:59:28)    转载▼    标签:    python   ...

  2. python中json格式数据输出实现方式

    python中json格式数据输出实现方式 主要使用json模块,直接导入import json即可. 小例子如下: #coding=UTF-8 import json info={} info[&q ...

  3. Python中变量的本质探索

    Python中变量的本质探索 参考:Vamei博客Python进阶09 动态类型 ''' a = [1,2,3] ''' (1)这条"赋值语句"实际上是将a指向对象"[1 ...

  4. python中星号变量的几种特殊用法

    python中星号变量的几种特殊用法 不知道大家知不知道在Python中,星号除了用于乘法数值运算和幂运算外,还有一种特殊的用法"在变量前添加单个星号或两个星号",实现多参数的传入 ...

  5. 【转】Python中执行cmd的三种方式

    原文链接:http://blog.csdn.net/menglei8625/article/details/7494094 目前我使用到的python中执行cmd的方式有三种: 1. 使用os.sys ...

  6. Python中字符串拼接的三种方式

    在Python中,我们经常会遇到字符串的拼接问题,在这里我总结了三种字符串的拼接方式:     1.使用加号(+)号进行拼接 加号(+)号拼接是我第一次学习Python常用的方法,我们只需要把我们要加 ...

  7. python中字符串的几种表达方式(用什么方式表示字符串)

    说明: 今天在学习python的基础的内容,学习在python中如何操作字符串,在此记录下. 主要是python中字符串的几种表达,表示方式. python的几种表达方式 1 使用单引号扩起来字符串 ...

  8. Python 中星号作用:解包&打散

    python中’*’和’**’的使用分两个方面,一个是计算,另一个是参数传递过程中元素的打包和解包.  计算方面 ‘*’和’**’在python中最常见的作用分别是‘相乘’和‘乘幂’,如下: > ...

  9. python中字符串的四种表达方式

    今天在学习python的基础的内容,学习在python中如何操作字符串,在此记录下. 主要是python中字符串的几种表达,表示方式. python的几种表达方式 1 使用单引号扩起来字符串 > ...

随机推荐

  1. sql server 高可用故障转移(3)

    虚拟磁盘创建 前面我们已经搭了域和两台sql 服务器, 下面我们准备让DC域服务器除了担当域控制器外,还行使另一个职能:充当集群共享存储. 集群共享存储是由群集内的每个节点都能共同访问的一个存储设备, ...

  2. SQL注入简单介绍

    一.SQL注入概念   1.sql注入是一种将sql代码添加到输入参数中   2.传递到sql服务器解析并执行的一种攻击手法   举例:某个网站的用户名为name=‘admin’.执行时为select ...

  3. struts2(二)---ModelDriven模型驱动

    这篇文章是在上一篇文章(http://blog.csdn.net/u012116457/article/details/48194905)的基础上写的,大家可以先快速阅读一下上一篇. 这篇文章用来写一 ...

  4. java的Junit的用法(转发)

    初级https://blog.csdn.net/andycpp/article/details/1327147/ 中级https://blog.csdn.net/andycpp/article/det ...

  5. bzoj 1189 紧急疏散 网络流

    二分答案,网络流判断 将每个门拆点,每个人连向每个门的dis~当前解 然后跑最大流,如果等于人数,即为可行解 #include<cstdio> #include<iostream&g ...

  6. C#进度框

    1.方法一:使用线程 功能描述:在用c#做WinFrom开发的过程中.我们经常需要用到进度条(ProgressBar)用于显示进度信息.这时候我们可能就需要用到多线程,如果不采用多线程控制进度条,窗口 ...

  7. Spring事务(一) Markdown 版

    事务 事务的特性(ACID) 原子性(Atomicity) 原子性要求事务所包含的全部操作是一个不可分割的整体,这些操作要么全部提交成功,要么只要其中一个操作失败,就全部"成仁" ...

  8. h5仿微信聊天(高仿版)、微信聊天表情|对话框|编辑器

    之前做过一版h5微信聊天移动端,这段时间闲来无事就整理了下之前项目,又重新在原先的那版基础上升级了下,如是就有了现在的h5仿微信聊天高仿版,新增了微聊.通讯录.探索.我四个模块 左右触摸滑屏切换,聊天 ...

  9. RabbitMQ+Spring 结合使用

    1:创建一个Maven工程,pom.xml文件如下: <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi ...

  10. python 编译源文件

    背景 近期项目到了部署的阶段.由于项目后台和算法都是用Python "撸的",但是又不希望将源代码直接 "release" 到 “客户”哪里.于是开始思考... ...