*args 和 **kwargs 的使用 [重复]

2024-11-19 08:38:00
admin
原创
9
摘要:问题描述:*args所以我对和的概念感到困难**kwargs。到目前为止我了解到:*args= 参数列表 - 作为位置参数**kwargs= 字典 - 其键成为单独的关键字参数,其值成为这些参数的值。我不明白这对什么编程任务有帮助。或许:我认为将列表和字典作为函数的参数同时作为通配符输入,这样我就可以传递任何...

问题描述:

*args所以我对和的概念感到困难**kwargs

到目前为止我了解到:

  • *args= 参数列表 - 作为位置参数

  • **kwargs= 字典 - 其键成为单独的关键字参数,其值成为这些参数的值。

我不明白这对什么编程任务有帮助。

或许:

我认为将列表和字典作为函数的参数同时作为通配符输入,这样我就可以传递任何参数?

有没有一个简单的例子来解释如何*args使用**kwargs

另外我发现的教程只使用了“*”和一个变量名。

*args和只是占位符吗,还是您在代码中**kwargs确实使用*args和?**kwargs


解决方案 1:

语法是*and**。名称*args**kwargs只是惯例,并没有强制要求使用它们。

*args当您不确定要向函数传递多少个参数时,可以使用它,即它允许您向函数传递任意数量的参数。例如:

>>> def print_everything(*args):
        for count, thing in enumerate(args):
...         print( '{0}. {1}'.format(count, thing))
...
>>> print_everything('apple', 'banana', 'cabbage')
0. apple
1. banana
2. cabbage

类似地,**kwargs允许您处理未事先定义的命名参数:

>>> def table_things(**kwargs):
...     for name, value in kwargs.items():
...         print( '{0} = {1}'.format(name, value))
...
>>> table_things(apple = 'fruit', cabbage = 'vegetable')
cabbage = vegetable
apple = fruit

您也可以将它们与命名参数一起使用。显式参数首先获取值,然后将其他所有内容传递给*argsand **kwargs。命名参数排在列表中的第一位。例如:

def table_things(titlestring, **kwargs)

您也可以在同一个函数定义中同时使用两者,但*args必须在之前出现**kwargs

调用函数时也可以使用*and**语法。例如:

>>> def print_three_things(a, b, c):
...     print( 'a = {0}, b = {1}, c = {2}'.format(a,b,c))
...
>>> mylist = ['aardvark', 'baboon', 'cat']
>>> print_three_things(*mylist)
a = aardvark, b = baboon, c = cat

正如您在本例中看到的,它获取项目列表(或元组)并将其解包。通过这种方式,它将它们与函数中的参数进行匹配。当然,您可以*在函数定义和函数调用中同时使用。

解决方案 2:

使用*argsand**kwargs非常有用的地方之一是用于子类化。

class Foo(object):
    def __init__(self, value1, value2):
        # do something with the values
        print value1, value2

class MyFoo(Foo):
    def __init__(self, *args, **kwargs):
        # do something else, don't care about the args
        print 'myfoo'
        super(MyFoo, self).__init__(*args, **kwargs)

这样,您就可以扩展 Foo 类的行为,而不必了解太多有关 Foo 的知识。如果您正在针对可能会更改的 API 进行编程,这会非常方便。MyFoo 只需将所有参数传递给 Foo 类。

解决方案 3:

这是一个使用 3 种不同类型参数的示例。

def func(required_arg, *args, **kwargs):
    # required_arg is a positional-only parameter.
    print required_arg

    # args is a tuple of positional arguments,
    # because the parameter name has * prepended.
    if args: # If args is not empty.
        print args

    # kwargs is a dictionary of keyword arguments,
    # because the parameter name has ** prepended.
    if kwargs: # If kwargs is not empty.
        print kwargs

>>> func()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: func() takes at least 1 argument (0 given)

>>> func("required argument")
required argument

>>> func("required argument", 1, 2, '3')
required argument
(1, 2, '3')

>>> func("required argument", 1, 2, '3', keyword1=4, keyword2="foo")
required argument
(1, 2, '3')
{'keyword2': 'foo', 'keyword1': 4}

解决方案 4:

这是我最喜欢使用该语法的地方之一,**如 Dave Webb 的最后例子中所示:

mynum = 1000
mystr = 'Hello World!'
print("{mystr} New-style formatting is {mynum}x more fun!".format(**locals()))

我不确定与仅使用名称本身相比它是否非常快,但输入起来要容易得多!

解决方案 5:

args 和 *kwargs 有用的一种情况是编写包装函数(例如装饰器)时,这些函数需要能够接受任意参数以传递给被包装的函数。例如,一个简单的装饰器,它打印被包装函数的参数和返回值:

def mydecorator( f ):
   @functools.wraps( f )
   def wrapper( *args, **kwargs ):
      print "Calling f", args, kwargs
      v = f( *args, **kwargs )
      print "f returned", v
      return v
   return wrapper

解决方案 6:

args 和 *kwargs 是 Python 的特殊魔法功能。想象一下一个可能具有未知数量参数的函数。例如,无论出于什么原因,您都希望拥有一个可以对未知数量的数字求和的函数(并且您不想使用内置的 sum 函数)。因此,您编写了这个函数:

def sumFunction(*args):
  result = 0
  for x in args:
    result += x
  return result

并像这样使用它:sumFunction(3,4,6,3,6,8,9)。

kwargs 具有不同的功能。使用 kwargs,您可以为函数提供任意关键字参数,并且可以将它们作为字典访问。

def someFunction(**kwargs):
  if 'text' in kwargs:
    print kwargs['text']

调用someFunction(text="foo")将打印foo。

解决方案 7:

假设你有一个函数,但你不想限制它所采用的参数数量。例如:

>>> import operator
>>> def multiply(*args):
...  return reduce(operator.mul, args)

然后你可以像这样使用这个函数:

>>> multiply(1,2,3)
6

or

>>> numbers = [1,2,3]
>>> multiply(*numbers)
6

解决方案 8:

名称*args**kwargs**kw纯粹是惯例。这让我们更容易阅读彼此的代码

一个方便的地方是使用 struct 模块时

struct.unpack()返回一个元组,而struct.pack()使用可变数量的参数。当操作数据时,能够将元组传递给struck.pack()例如会很方便。

tuple_of_data = struct.unpack(format_str, data)
# ... manipulate the data
new_data = struct.pack(format_str, *tuple_of_data)

如果没有这个能力,你就不得不写

new_data = struct.pack(format_str, tuple_of_data[0], tuple_of_data[1], tuple_of_data[2],...)

这也意味着如果 format_str 发生改变并且元组的大小也发生改变,我就必须返回并编辑那行很长的代码

解决方案 9:

请注意,args/kwargs 是函数调用语法的一部分,而不是真正的运算符。我遇到了一个特殊的副作用,即您不能将 args 扩展与 print 语句一起使用,因为 print 不是函数。

这似乎是合理的:

def myprint(*args):
    print *args

不幸的是它无法编译(语法错误)。

编译如下:

def myprint(*args):
    print args

但将参数打印为元组,这不是我们想要的。

这是我决定的解决方案:

def myprint(*args):
    for arg in args:
        print arg,
    print

解决方案 10:

这些参数通常用于代理函数,因此代理可以将任何输入参数传递给目标函数。

def foo(bar=2, baz=5):
    print bar, baz

def proxy(x, *args, **kwargs): # reqire parameter x and accept any number of additional arguments
    print x
    foo(*args, **kwargs) # applies the "non-x" parameter to foo

proxy(23, 5, baz='foo') # calls foo with bar=5 and baz=foo
proxy(6)# calls foo with its default arguments
proxy(7, bar='asdas') # calls foo with bar='asdas' and leave baz default argument

但由于这些参数隐藏了实际的参数名称,因此最好避免使用它们。

解决方案 11:

您可以查看 python 文档(docs.python.org 中的常见问题解答),但更具体地说,为了更好地解释神秘的 miss args 和 mister kwargs(由 archive.org 提供)(原始的无效链接在这里)。

简而言之,当函数或方法使用可选参数时,两者都会用到。正如 Dave 所说,当您不知道可以传递多少个参数时,使用 args,而当您想要处理按名称和值指定的参数时,使用 *kwargs,如下所示:

myfunction(myarg=1)
相关推荐
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   601  
  华为IPD与传统研发模式的8大差异在快速变化的商业环境中,产品研发模式的选择直接决定了企业的市场响应速度和竞争力。华为作为全球领先的通信技术解决方案供应商,其成功在很大程度上得益于对产品研发模式的持续创新。华为引入并深度定制的集成产品开发(IPD)体系,相较于传统的研发模式,展现出了显著的差异和优势。本文将详细探讨华为...
IPD流程是谁发明的   7  
  如何通过IPD流程缩短产品上市时间?在快速变化的市场环境中,产品上市时间成为企业竞争力的关键因素之一。集成产品开发(IPD, Integrated Product Development)作为一种先进的产品研发管理方法,通过其结构化的流程设计和跨部门协作机制,显著缩短了产品上市时间,提高了市场响应速度。本文将深入探讨如...
华为IPD流程   9  
  在项目管理领域,IPD(Integrated Product Development,集成产品开发)流程图是连接创意、设计与市场成功的桥梁。它不仅是一个视觉工具,更是一种战略思维方式的体现,帮助团队高效协同,确保产品按时、按质、按量推向市场。尽管IPD流程图可能初看之下显得错综复杂,但只需掌握几个关键点,你便能轻松驾驭...
IPD开发流程管理   8  
  在项目管理领域,集成产品开发(IPD)流程被视为提升产品上市速度、增强团队协作与创新能力的重要工具。然而,尽管IPD流程拥有诸多优势,其实施过程中仍可能遭遇多种挑战,导致项目失败。本文旨在深入探讨八个常见的IPD流程失败原因,并提出相应的解决方法,以帮助项目管理者规避风险,确保项目成功。缺乏明确的项目目标与战略对齐IP...
IPD流程图   8  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用