python中前缀运算符 *和 **的用法示例详解


Posted in Python onMay 28, 2020

这篇主要探讨 ** 和 * 前缀运算符,**在变量之前使用的*and **运算符.

一个星(*):表示接收的参数作为元组来处理

两个星(**):表示接收的参数作为字典来处理

简单示例:

>>> numbers = [2, 1, 3, 4, 7]
>>> more_numbers = [*numbers, 11, 18]
>>> print(*more_numbers, sep=', ')
2, 1, 3, 4, 7, 11, 18

用途:

  • 使用 * 和 ** 将参数传递给函数
  • 使用**和**捕获传递给函数的参数
  • 使用*只接受关键字参数
  • 使用*元组拆包过程中捕获项目
  • 使用*解包iterables到一个列表/元组
  • 使用**要解压缩词典到其他字典

例子解释:

1.调用函数时,*可以使用运算符将​​可迭代对象解压缩为函数调用中的参数:

>>> 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作为单独的参数传递到函数调用中,而我们甚至不需要知道列表中有多少个参数。

2.** 运算符允许我们采取键值对的字典,并把它解压到函数调用中的关键字参数。

>>> date_info = {'year': "2020", 'month': "01", 'day': "01"}
>>> filename = "{year}-{month}-{day}.txt".format(**date_info)
>>> filename
'2020-01-01.txt'

** 将关键字参数解包到函数调用中并不是很常见。我最常看到的地方是练习继承时:super()通常要同时包含*和**。
双方*并 **可以在函数调用中多次使用,像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'

3.定义函数时,*可以使用运算符捕获为函数提供的无限数量的位置参数。这些参数被捕获到一个元组中。

from random import randint

def roll(*dice):
  return sum(randint(1, die) for die in dice

4.我们可以用**定义一个函数时,捕捉给予功能到字典中的任何关键字参数:

def tag(tag_name, **attributes):
  attribute_list = [
    f'{name}="{value}"'
    for name, value in attributes.items()
  ]
  return f"<{tag_name} {' '.join(attribute_list)}>"

5.带有仅关键字参数的位置参数,要接受仅关键字的参数,可以*在定义函数时在使用后放置命名参数

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']

参数dictionaryand default在其后*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'

6.不带位置参数的仅关键字参数
仅关键字参数的功能很酷,但是如果您需要仅关键字参数而不捕获无限的位置参数怎么办?

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
  ```

该函数接受一个iterable参数,该参数可以在位置上指定(作为第一个参数),也可以通过其名称和作为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 必须指定为关键字arguments。

7.元组拆包中的星号

Python 3还添加了一种新的使用运算符的方式,该方式仅与上面的-when-defining-a-function和*-when-when-calling-afunction功能有关。

>>> 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']

8.列表文字中的星号
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'}

请注意,上面的最后一行获取一个列表和一个生成器,并将它们转储到新集中。在使用之前*,以前没有一种简单的方法可以在一行代码中做到这一点。以前有一种方法可以做到,但要记住或发现它并不容易:

>>> set().union(fruits, uppercase_fruits)
{'lemon', 'watermelon', 'TOMATO', 'LEMON', 'PEAR', 'WATERMELON', 'tomato', 'pear'}

9.字典文字中的双星号
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'}

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

>>> 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'}

ref: https://treyhunner.com/2018/10/asterisks-in-python-what-they-are-and-how-to-use-them/

总结

到此这篇关于python中前缀运算符 *和 **的用法示例详解的文章就介绍到这了,更多相关python中 *和 **的用法内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章希望大家以后多多支持三水点靠木!

Python 相关文章推荐
python实现的简单FTP上传下载文件实例
Jun 30 Python
Python通过命令开启http.server服务器的方法
Nov 04 Python
pip安装python库的方法总结
Aug 02 Python
Django之编辑时根据条件跳转回原页面的方法
Aug 21 Python
python实现文件的分割与合并
Aug 29 Python
Python OrderedDict的使用案例解析
Oct 25 Python
python使用HTMLTestRunner导出饼图分析报告的方法
Dec 30 Python
Python 将json序列化后的字符串转换成字典(推荐)
Jan 06 Python
python数据类型可变不可变知识点总结
Mar 06 Python
关于keras.layers.Conv1D的kernel_size参数使用介绍
May 22 Python
python hmac模块验证客户端的合法性
Nov 07 Python
python中os.remove()用法及注意事项
Jan 31 Python
PHP基于phpqrcode类库生成二维码过程解析
May 28 #Python
Python函数参数分类原理详解
May 28 #Python
pygame用blit()实现动画效果的示例代码
May 28 #Python
PyCharm中如何直接使用Anaconda已安装的库
May 28 #Python
Python内置异常类型全面汇总
May 28 #Python
python不到50行代码完成了多张excel合并的实现示例
May 28 #Python
python使用openpyxl操作excel的方法步骤
May 28 #Python
You might like
php jquery 实现新闻标签分类与无刷新分页
2009/12/18 PHP
单台服务器的PHP进程之间实现共享内存的方法
2014/06/13 PHP
ThinkPHP类似AOP思想的参数验证的实现方法
2019/12/18 PHP
JS 控件事件小结
2012/10/31 Javascript
一个简单的JS鼠标悬停特效具体方法
2013/06/17 Javascript
SuperSlide2实现图片滚动特效
2014/06/20 Javascript
JavaScript实现的简单拖拽效果
2015/06/01 Javascript
jQuery实现内容定时切换效果完整实例
2016/04/06 Javascript
浅析JavaScript中浏览器的兼容问题
2016/04/19 Javascript
jQuery插件扩展实例【添加回调函数】
2016/11/26 Javascript
bootstrap组件之导航组件使用方法
2017/01/19 Javascript
BootStrap select2 动态改变值的方法
2017/02/10 Javascript
input输入框内容实时监测(附代码)
2017/08/15 Javascript
详解bootstrap用dropdown-menu实现上下文菜单
2017/09/22 Javascript
echarts整合多个类似option的方法实例
2018/07/10 Javascript
layui-laydate时间日历控件使用方法详解
2018/11/15 Javascript
vue自定义指令实现仅支持输入数字和浮点型的示例
2019/10/30 Javascript
vuex管理状态仓库使用详解
2020/07/29 Javascript
逐行分析鸿蒙系统的 JavaScript 框架(推荐)
2020/09/17 Javascript
Python中的二叉树查找算法模块使用指南
2014/07/04 Python
用PyQt进行Python图形界面的程序的开发的入门指引
2015/04/14 Python
python回调函数中使用多线程的方法
2017/12/25 Python
python 3.7.0 安装配置方法图文教程
2018/08/27 Python
Python中extend和append的区别讲解
2019/01/24 Python
在Python中合并字典模块ChainMap的隐藏坑【推荐】
2019/06/27 Python
使用Python和Scribus创建一个RGB立方体的方法
2019/07/17 Python
html5 Canvas画图教程(3)—canvas出现1像素线条模糊不清的原因
2013/01/09 HTML / CSS
Vilebrequin欧洲官网:法国豪华泳装品牌(男士沙滩裤)
2018/04/14 全球购物
彪马香港官方网上商店:PUMA香港
2020/12/06 全球购物
企业法人授权委托书范本
2014/09/23 职场文书
2014院党委领导班子对照检查材料思想汇报
2014/09/24 职场文书
六查六看自查报告
2014/10/14 职场文书
蓬莱阁导游词
2015/02/04 职场文书
高中生思想道德自我评价
2015/03/09 职场文书
2016入党积极分子党课培训心得体会
2016/01/06 职场文书
员工安全责任协议书
2016/03/22 职场文书