Python 在形参和实参中使用 * 和 **

一、在形参中使用***是比较常见的写法

形参中使用***的原理可以参考《 * 与 ** 参数问题》这篇文章,讲解非常详细,本文只作一个学习总结。

  1. *args是可变参数,args 接收的是一个元组。比如可以向一个函数传入一个列表(或迭代器),在形参中是使用*就可以在函数中依次接收这个列表的每一个元素;
  2. **kw是关键字参数,kw 接收的是一个字典。比如在收集用户信息的时候,除了必填项之外,我们可以用**来方便地接收那些未知个数的可选项。
  3. 在 lambda 表达式中使用***
    lambda 表达式可以当做一个匿名函数,是函数都可以传参,匿名函数的形参中也带有使用***
  a = lambda *s: s + (2, 'g')
  print(a('dd', 'de'))

  # 打印出来的结果:('dd', 'de', 2, 'g')
  lambda_test = lambda **args: args
  print(lambda_test(usr='Obama', age=18))

  # 打印出来的结果:{'usr': 'Obama', 'age': 18}

二、当对象为迭代器时,可在实参中使用*;当对象为一个可迭代的映射时,可在实参中使用**

众所周知,Python 中的可迭代对象分为集合数据类型(list、tuple、dict、set、str)和 generator 两种。
而我们常见的可迭代映射就是 dict 啦。

  1. 实参中的*会获取可迭代对象的 key(索引/键值)
    由于获取的是可迭代对象的 key,所以函数的形参可以为位置参数或可变参数。
  • 形参为位置参数
    def foo(n1, n2, n3, n4, n5):
        return n1, n2, n3, n4, n5

    d = {'n1': 5, 'n2': 6, 'n3': 7, 'n4': 8, 'n5': 9}
    print(foo(*d))  

    # 打印得到的结果为 ('n1', 'n2', 'n3', 'n4', 'n5')
  • 形参为可变参数
    def fu(*n):
        print(n)

    d = {'n1': 5, 'n2': 6, 'n3': 7, 'n4': 8, 'n5': 9}
    print(fu(*d))

    # 打印得到的结果为 ('n1', 'n2', 'n3', 'n4', 'n5')

以上两个例子中,将实参中的字典对象换成列表、元组或者字符串对象,都是可以的,朋友们可以自己动手尝试一下。

  1. 实参中的**会获取可迭代映射的 value
    由于获取的是可迭代对象的 value,所以函数的形参只能为关键字参数,即参数的个数和名称都是固定的。
    def foo(n1, n2, n3, n4, n5):
        return n1, n2, n3, n4, n5

    d = {'n1': 5, 'n2': 6, 'n3': 7, 'n5': 8, 'n4': 9}
    print(foo(**d))

    # 打印得到的结果为(5, 6, 7, 9, 8)

如果我们要调用的函数不是关键字形参:

    def fu(*n):
        print(n)
        
    iterator = 'unique'
    print(fu(**iterator))
    # 会报错为:TypeError: fu() argument after ** must be a mapping

如果调用函数中的关键字形参名称与传入的实参无法一一对应:

    def foo(n1, n2, n3, n4, n5):
        return n1, n2, n3, n4, n5

    d = {'n1': 5, 'n2': 6, 'n3': 7, 'n4': 8, 'n5': 9, 'n6': 10}
    print(foo(**d))
    # 会报错为:TypeError: foo() got an unexpected keyword argument 'n6'

更多举例说明请参考《Python 函数参数类型 * 的区别》这篇文章。

你可能感兴趣的:(Python 在形参和实参中使用 * 和 **)