1.参数解包:方法调用中的*表达式**表达式

如果语法*表达式出现在函数调用中,则该表达式必须是可迭代的。

这些可迭代集合的元素被视为附加的位置参数。

对于调用f(x1, x2, *y, x3, x4),如果y等于序列[y1,...,yM],则等效于调用f(x1, x2, y1, ..., yM, x3, x4)

如果函数调用中出现语法**表达式,则该表达式的值必须为“映射”,其内容被视为附加的关键字参数。

如果一个关键字已经存在(作为一个明确的关键字参数,或来自另一个解包),则会引发“ TypeError”异常。

关于参数顺序的说明:* 表达式可以出现在显式关键字参数之后,并在关键字参数和任何** 表达式参数之前被处理。而在一般函数的显式参数调用中,位置参数必须放在关键字参数之前。例如:

def func(a, b ,c):
print(a, b, c) func(c=3, 1, 2) # SyntaxError: positional argument follows keyword argument
func(c=3, *(1,2)) # OK. 1 2 3
func(c=3, **dict(a=1,b=2)) # OK. 1 2 3
func(c=3, *(1,), **dict(b=2)) # OK. 1 2 3

2.参数解包:方法定义中的*表达式**表达式

如果存在形式 * identifier,则将其初始化为接收任何其余位置参数的元组,默认为空元组。

如果存在形式 ** identifier,则将其初始化为一个新字典,该字典将接收任何其余的关键字参数,默认为一个新的空字典。

所以 *args其实是一个元组!

让我们来看一个例子:

def func(*args, **kwargs):
print(args)
print(kwargs)
print(locals()) func(1, a=2) # output:
# (1,)
# {'a': 2}
# {'kwargs': {'a': 2}, 'args': (1,)}

单元素元组的语法看起来有点奇怪。不过习惯就好了。

仅需使用尾部逗号就能创建单个元组(也称为单例);在所有其他情况下,逗号是可选的。没有尾部逗号的单个表达式不会创建元组,而是会产生该表达式的值。

3.在元组,列表,集合和字典中解包

>>> *range(4), 4
(0, 1, 2, 3, 4)
>>> [*range(4), 4]
[0, 1, 2, 3, 4]
>>> {*range(4), 4}
{0, 1, 2, 3, 4}
>>> {'x': 1, **{'y': 2}}
{'x': 1, 'y': 2}

在字典中,后一个值将始终覆盖前一个值。

  • 注意,我们也可以这样理解

    • *可迭代集合解包算子
    • **字典解包算子

4.Extended Unpacking:赋值表达式左边的*表达式

赋值表达式左侧的元组(或列表)至多可以包含一个带有单个星号的表达式(称为“星号”表达式,而列表中的其他表达式则称为“强制性”)。

强制表达式的值将根据赋值表达式右边元组(或列表)中对应的元素位置被分配。

例如,如果seq是可切分的序列,且seq具有至少2个元素,则以下所有赋值均等效:

a, *b, c = seq
[a, *b, c] = seq
a, b, c = seq[0], list(seq[1:-1]), seq[-1]
  • seq[0]保证赋值给a
  • seq[-1]保证赋值给c
  • 其余元素则赋值给b,b一定是个列表。
  • 如果 len(seq)==2,b为空列表

如果星号表达式赋值的长列表,那么会报错:

*a = range(5)  # SyntaxError: starred assignment target must be in a list or tuple

这样就可以了:

*a, = range(5)  # OK
a: [0, 1, 2, 3, 4]

该建议也适用于隐式赋值中的元组,例如在for语句中:

for a, *b in [(1, 2, 3), (4, 5, 6, 7)]:
print(b) # output:
# [2, 3]
# [5, 6, 7]

更多的例子可以参考 stack overflow: Unpacking, Extended unpacking, and nested extended unpacking

Ref:

http://yaoyao.codes/python/2016/09/25/python-starred-expression

The Python Language Reference - 6.3.4. Calls

The Python Language Reference - 8.6. Function definitions

The Python Language Reference - 6.14. Expression lists

PEP 448 – Additional Unpacking Generalizations

PEP 3132 – Extended Iterable Unpacking

Python中的参数解包:`*`表达式和 `**`表达式的更多相关文章

  1. python之参数解包

    # 参数解包:将整个list当做参数传给函数 list = [1, 2, 4] def add_fn(a, b, c): return a + b + c sum = add_fn(*list) pr ...

  2. Python中的参数

    Python中的参数 1. python函数参数有多重形式: * test(arg1,arg2,`*args`) * test(arg1,arg2,`*args`,`**kwargs`) 2. 其中比 ...

  3. python基础之打/解包及运算符与控制流程

    python基础之打/解包及运算符与控制流程 python中的解压缩(即序列类型的打包和解包) python提供了两个设计元祖和其他序列类型的处理的便利,也就是自动打包与自动解包功能,比如: data ...

  4. python中常用的导包的方法和常用的库

    python中常用的导包的方法               导入包和包名的方法:1.import package.module 2.from package.module import  * 例一: ...

  5. Python中dict详解

    from:http://www.cnblogs.com/yangyongzhi/archive/2012/09/17/2688326.html Python中dict详解 python3.0以上,pr ...

  6. 从 python 中 axis 参数直觉解释 到 CNN 中 BatchNorm 的工作方式(Keras代码示意)

    1. python 中 axis 参数直觉解释 网络上的解释很多,有的还带图带箭头.但在高维下是画不出什么箭头的.这里阐述了 axis 参数最简洁的解释. 假设我们有矩阵a, 它的shape是(4, ...

  7. (转)python中函数参数中如果带有默认参数list的特殊情况

    在python中函数参数中如果带有默认参数list遇到问题 先看一段代码 1 2 3 4 5 6 7 8 9 def f(x,l=[]):     for i in range(x):         ...

  8. python中函数参数的引用方式

    值传递和引用传递时C++中的概念,在python中函数参数的传递是变量指向的对象的物理内存地址!!! python不允许程序员选择采用传值还是传引用.Python参数传递采用的肯定是“传对象引用”的方 ...

  9. Python 中的时间处理包datetime和arrow

    Python 中的时间处理包datetime和arrow 在获取贝壳分的时候用到了时间处理函数,想要获取上个月时间包括年.月.日等 # 方法一: today = datetime.date.today ...

随机推荐

  1. 白面系列 kafka

    kafka是一个分布式发布订阅消息系统,也可叫做MQ系统,MQ是Message Queue,消息队列. 通俗点,生产者往队列里写消息,消费者从队列里读.专业点,Producer通过TCP协议发送消息到 ...

  2. 转: Struts2中拦截器与过滤器的区别及执行顺序

    当接收到一个httprequest , a) 当外部的httpservletrequest到来时 b) 初始到了servlet容器 传递给一个标准的过滤器链 c) FilterDispatecher会 ...

  3. python 数组格式转换

    格式转换 arr1 = [ {'name': 'jack', 'hobby': '西瓜'}, {'name': 'jack', 'hobby': '冬瓜'}, {'name': 'rose', 'ho ...

  4. php中构建树状图

    /** * 指定根层级的树状图 * @param array $list 初始数组 * @param int $root 最上级一条数据的id * @param string $pk 每一条数据的id ...

  5. SGD 讲解,梯度下降的做法,随机性。理解反向传播

    SGD 讲解,梯度下降的做法,随机性.理解反向传播 待办 Stochastic Gradient Descent 随机梯度下降没有用Random这个词,因为它不是完全的随机,而是服从一定的分布的,只是 ...

  6. pl/sql developer试用期结束

    (产品代码)Product Code:ljkfuhjpccxt8xq2re37n97595ldmv9kch (序列号)Serial Number:302967 (口令)Password:xs374ca ...

  7. 基于alpine的php-fpm扩展swoole和pdo_mysql

    vim Dockerfile 插入一下内容 FROM php:fpm-alpine RUN echo http://mirrors.aliyun.com/alpine/v3.10/main>/e ...

  8. 跨站跟踪攻击(CST/XST)

    XSS与httponly 正常情况下,客户端脚本(如JS脚本)是可以通过document.cookie函数获得,这样如果有XSS跨站漏洞,cookie很容易被盗取.浏览器有一个安全策略,通过设置coo ...

  9. flask入门(二)

    接着上文 讲一讲响应 flask调用视图函数后,会将其返回值作为响应的内容.大多数情况下,响应就是一个简单的字符串,作为HTML页面回送客户端.但HTTP协议需要的不仅是作为请求响应的字符串.HTTP ...

  10. C++ 中字符串查找、字符串截取、字符串替换

    参照:C++基础-string截取.替换.查找子串函数 1.字符串查找 s.find(s1) //查找s中第一次出现s1的位置,并返回(包括0) s.rfind(s1) //查找s中最后次出现s1的位 ...