Python 语言中 asterisk 符号用法小结
对于 asterisk 符号的使用,本文分为预定义(predefined)和自定义(customized)两部分进行介绍。预定义部分,是指 Python 语言中已经预先定义好的部分,直接用就可以了;自定义,是指定制的,需要程序员自定义相应的功能。注意,这里讨论的 asterisk 符号的使用包括 * 符号和 ** 符号两种类型。
一、预定义
下面介绍 asterisk 符号在数值类型、非数值内置类型、函数定义和赋值中的用法。
1. 数值类型中 asterisk 符号的用法
在整型和浮点型等数值类型中,* 符号作为乘法运算符,** 符号作为乘方运算符。
具体的示例程序,如下:
x, y = 2, 4
res1 = x * y # 8
res2 = x ** y # 16
res1 *= x # 16
res2 **= x # 256
2. 非数值内置类型中 asterisk 符号的用法
所有的非数值内置类型都不支持 ** 符号;而对于 * 符号,只有 tuple 或 list 类型支持 * 符号操作符,而 set 或 dict 类型不支持,具体如下:
元组(tuple)或列表(list)类型,* 符号可使非数值类型变量重复指定次数;
而集合(set)和字典类型(dict)是不支持 * 符号操作符的,会抛出 "TypeError: unsupported operand type(s)" 异常。具体示例如下:
# list examples
print([] * 3) # []
print(list() * 3) # []
print([0] * 3) # [0, 0, 0]
# tuple examples
print(() * 3) # ()
print(tuple() * 3) # ()
print((0,) * 3) # (0, 0, 0)
# set examples, throw TypeError Exception
try:
set() * 3
set([]) * 3
{0} * 3
except TypeError as e:
print(e)
pass
# dict examples, throw TypeError Exception
try:
{} * 3
dict() * 3
{0: '0'} * 3
except TypeError as e:
print(e)
pass
注意,(0,) 是元组类型,而(0)是整型,不是元组类型,区别只在于缺少了一个逗号(,)。
3. asterisk 符号在函数定义中的用法
在 Python 函数定义中,* 符号表示允许函数接受数量可变的参数(arguments);** 符号表示允许函数接受数量可变的关键字参数(keyword arguments)。具体示例如下:
def func(*args, **kwargs):
print('args:', args)
print('kwargs:', kwargs)
func((0,1,2), 3, 4, 5, key=0, value='0')
# args: ((0, 1, 2), 3, 4, 5)
# kwargs: {'key': 0, 'value': '0'}
注意,在 Python 函数定义中,带有 * 符号参数,必须在带有 ** 符号参数之前。
在 Python 3 中,有一个新用法,称为 bare asterisk/star。即,在函数定义中,参数之间,有一个光秃秃的 * 符号。这种用法的含义是 bare star 的后面,只能接受关键字参数。更多解释参见 PEP 3102 -- Keyword-Only Arguments。具体示例如下:
def compare1(x, y, *, key=None):
print('x:', x)
print('y:', y)
print('key:', key)
return x > y
def compare2(x, y, *, z, key=None):
print('x:', x)
print('y:', y)
print('key:', key)
if z < 0: z *= -1
return z if x > y else -1
compare1(3, 5)
compare2(3, 5, key=2, z=2)
try:
compare2(3, 5, 2)
except TypeError as e:
print(e)
4. asterisk 符号在赋值中的用法
在调用函数时,采用 * 符号可以拆解一些内置类型,使之变成一个个单独的元素再传入函数中。具体示例如下:
def func(*args):
print('args:', args)
params = ['RGBA', (1,2,3), [1,2,3], {1,2,3}]
# call function with arguments(* included)
print("\ncall function with arguments(* included)")
for args in params:
func(*args)
# call function with arguments(* not included)
print("\ncall function with arguments(* not included)")
for args in params:
func(args)
在 Python 3 中,还可以把带有 * 符号的操作数(operand),或者说是变量,放到赋值符号的左侧,从而扩展了迭代器(iterator)的拆解赋值。具体示例如下:
iterators = [(0,1,2,3,4), 'ABCDE', [0,1,2,3,4], {0,1,2,3,4}]
for item in iterators:
x, *y, z = item
print('\nitem:', item)
print('x: %s, y: %s, z: %s' % (x,y,z))
print('-' * 50)
二、自定义
对于自定义的类,可以通过定义相应的魔法方法(magic method),实现asterisk符号运算符(operator)。asterisk 符号相关的操作符与魔法方法的具体对应关系,如表1所示。
操作符 | 魔法方法 | 原始的表现形式 | 解释 |
* | __mul__(self, other) | self * other | 前向(forward)相乘方法 |
* | __rmul__(self, other) | other * self | 反转(reverse)相乘方法 |
*= | __imul__(self, other) | self *= other | 原地(in-place)相乘方法 |
** | __pow__(self, other[, modulo]) | self ** other | 前向乘方方法 |
** | __rpow__(self, other[, modulo]) | other ** self | 反转乘方方法 |
**= | __ipow__(self, other[, modulo]) | self **= other | 原地乘方方法 |
注,反转相乘方法和反转乘方方法只有在self实例没有定义对应的前向方法时调用;modulo 为可选参数,表示对 modulo 参数取模。
下面定义了一个 Length 类,实现将长度的单位都转换为米(meter)后,再进行相乘或乘方运算。具体示例代码,如下:
class Length(object):
__metric = {"mm": 0.001, "cm": 0.01, "m": 1, "km": 1000,
"in": 0.0254, "ft": 0.3048, "yd": 0.9144,
"mi": 1609.344}
def __init__(self, value, unit="m"):
self.value = value
self.unit = unit
def convert2meters(self):
return self.value * Length.__metric[self.unit]
def __str__(self):
"""Informal string representation for a user"""
return str(self.convert2meters())
def __repr__(self):
"""Official/Formal string representation for a programmer"""
return "Length(" + str(self.value) + ", '" + self.unit + "')"
def __add__(self, other):
meters = self.convert2meters() + other.convert2meters()
return Length(meters/Length.__metric[self.unit], self.unit)
def __mul__(self, other):
"""Regard the other object as a multiplier upon self"""
meters = self.convert2meters() * other.convert2meters()
return Length(meters/Length.__metric[self.unit], self.unit)
def __imul__(self, other):
"""In-place multiplication"""
meters = self.convert2meters() * other.convert2meters()
self.value = meters/Length.__metric[self.unit]
return self
def __pow__(self, other):
"""Regard the other object as an exponent upon self"""
meters = self.convert2meters() ** other.convert2meters()
return Length(meters/Length.__metric[self.unit], self.unit)
def __ipow__(self, other):
"""In-place power"""
meters = self.convert2meters() ** other.convert2meters()
self.value = meters/Length.__metric[self.unit]
return self
if __name__ == '__main__':
x = Length(4)
print(x)
y = eval(repr(x))
# Test * symbol operator
z1 = Length(4.5, "yd") * Length(1)
z1 *= x
print(repr(z1))
print(z1)
# Test ** operator
z2 = y ** x
z2 **= Length(1)
print(repr(z2))
print(z2)
参考资料
[1] Numeric Type Special Methods in Chapter 24. Creating or Extending Data Types.
https://www.linuxtopia.org/online_books/programming_books/python_programming/python_ch24s04.html
[2] Magic Methods and Operator Overloading. https://www.python-course.eu/python3_magic_methods.php