Python基础-*args和**kwargs魔法变量

在学习Python时,总会遇到*args和**kwargs这两个魔法变量,那么它们到底是什么?

首先,并不是必须写成*args和**kwargs。只有变量前面的*(星号)才是必须的,你也可以写成*var 和**vars,而写成*args 和**kwargs只是一个通俗的命名约定。 

1 *args的用法

*args和**kwargs主要用于函数定义。你可以将不定数量的参数传递给一个函数。

这里的不定的意思是:预先并不知道函数使用者会传递多少个参数给你,所以在这个场景下使用这两个关键字。

  • *args是用来发送一个非键值对的可变数量的参数列表给一个函数。

   这里有个例子帮你理解这个概念:

def test_var_args(f_arg, *argv):
    print("first normal arg:", f_arg)
    for arg in argv:
        print("another arg through *argv:", arg)
    
test_var_args('google', 'python', 'java', 'c++')
('first normal arg:', 'google')
('another arg through *argv:', 'python')
('another arg through *argv:', 'java')
('another arg through *argv:', 'c++')

2 **kwargs的用法

  • **kwargs允许你将不定长度的键值对(字典),作为参数传递给一个函数。如果你想要在一个函数里处理带名字的参数,你应该使用**kwargs。       
def greet_me(**kwargs):
    for key, value in kwargs.items():
        print("{0} == {1}".format(key, value))
        
greet_me(name="google", website="www.google.com", level=1)
website == www.google.com
name == google
level == 1

现在你可以看出我们怎样在一个函数里,处理了一个键值对参数了。

这就是**kwargs的基础,而且你可以看出它有多么管用。接下来让我们谈谈,怎样使用*args和**kwargs来调用一个参数为列表或者字典的函数。

3 使用*args和**kwargs来调用函数

那现在我们将看到怎样使用*args和**kwargs来调用一个函数。假设你有这样一个函数

def test_args_kwargs(arg1, arg2, arg3):
        print("arg1:", arg1)
        print("arg2:", arg2)
        print("arg3:", arg3)

你可以使用*args或**kwargs来给这个函数传递参数。下面是怎样做:

# 首先使用*args:
args = ("two", 3, 5)
print "首先使用*args"
test_args_kwargs(*args)
print "
"

# 现在使用**kwargs:       
print "首先使用**kwargs" 
kwargs = {"arg3": 3, "arg2": "two", "arg1": 5}
test_args_kwargs(**kwargs)
首先使用*args
('arg1:', 'two')
('arg2:', 3)
('arg3:', 5)


首先使用**kwargs
('arg1:', 5)
('arg2:', 'two')
('arg3:', 3)

如果你想在函数里同时使用标准参数、*args和**kwargs这三种参数, 顺序是这样的:

some_func(fargs, *args, **kwargs)

4 什么时候使用它们?

  •  这还真的要看你的需求而定。最常见的用例是在写函数装饰器的时候。
  •  此外它也可以用来做猴子补丁(monkey patching)。猴子补丁的意思是在程序运行时(runtime)修改某些代码。
  •  举个例子,你有一个类,里面有个叫get_info的函数会调用一个API并返回相应的数据。如果想测试它,可以把API调用替换成一些测试数据。例如:
import someclass2
def get_info(self, *args):
        return "Test data"

someclass.get_info = get_info
原文地址:https://www.cnblogs.com/xiyuan2016/p/7458914.html