题目描述: 1、怎么理解不定长参数? 2、*args 和 **kwargs 是什么意思?为什么要使用它们?
答案要点如下: 1、函数参数可为分如下几种:必选参数、默认参数、可变参数、命名关键字参数和关键字参数
2、当我们在定义和调用一个函数时,如果包含所有的参数类型,则必须按照:必选参数、默认参数、可变参数、命名关键字参数和关键字参数的顺序。但是在实际开发中,不建议包含过多的参数类型,会影响代码的可读性
3、必选参数很简单,就是函数中必须要接受的参数
4、默认参数,即给参数赋一个默认值,我们在传递时,可省略对该参数的传值操作。如:
def print_test(title, msg="world"):
print(title, msg)
print_test("hello") # hello world
print_test("hello", "demon") # hello demon
print_test("hello", msg="demon") # hello demon
错误调用示例
print_test( msg="demon","hello") # 这样是不对的
5、可变参数,在定义函数是用 *args 来接受,其中 * 是规定的,args可用其他名称替换,但一般习惯用 args 来表示。可变参数在传入函数后,被封装成一个 tuple 来进行使用。所以我们在函数内部,可以通过操作 tuple 的方法来操作参数,示例如下:
def print_numbers(*args):
print(type(args)) # tuple
for n in args:
print(type(n)) # int`
print_numbers(1, 2, 3, 4)
6、如果在函数外已经得到一个 list 或者 tuple,想调用一个可变参数,也可以用 *+变量名 的形式进行调用,(这种用法有点类似 C语言 中的指针),示例如下:
def print_numbers(*args):
print(type(args)) # tuple
for n in args:
print(type(n)) # int
l = [1, 2, 3, 4]
print_numbers(*l) # *l,等价于 print_numbers(1, 2, 3, 4)
print_numbers(l) # 将 l 作为一个整体传入,这样函数接受到的其实只有一个参数,且参数类型为 list
7、关键字参数使用 kwargs 来标识,是规定,而kwargs可替换,它将不定长参数转换为 dict 传入函数。它用于扩展函数的功能。比如我们要实现用户注册,有必输项和非必输项,这些非必输项就可以用关键字参数来接受。示例如下:
def register(name, email, **kwargs):
print('name:%s, age:%s, others:%s', (name, email, kw))
register("demon", "1@1.com") # name:%s, age:%s, others:%s ('demon', '1@1.com', {})
register("demon", "1@1.com", addr="shanghai") # name:%s, age:%s, others:%s ('demon', '1@1.com', {'addr': 'shanghai'})
8、如果函数外已经获得一个 dict ,想进行函数调用,只需用 **+变量名 传入即可。并且 dict示例如下:
def register(name, email, **kwargs):
print('name:%s, age:%s, others:%s', (name, email, kw))
d = {"addr":"shanghai"}
register("demon", "1@1.com", **d)
d = {"name":"yrr", "email":"1@1.com", "addr":"shanghai"}
register(**d)
d = {"email":"yrr", "name":"1@1.com", "addr":"shanghai"}
register(**d)
9、命名关键字参数用于限制调用函数可传入的属性。这里的限制是假限制,因为仍然可以传入,只是函数体中对限制外的参数不作任何处理。命名关键字参数用一个 * 号分隔,* 后面的参数都被视为命名关键字参数。如:
def person(name, age, *, city, job):
print(name, age, city, job)
10、如果函数中已经有了一个可变参数的定义,后面的命名关键字参数则不需要再添加 * ,如:
def person(name, age, *args, city, job):
print(name, age, args, city, job)
11、关键字参数和命名关键字参数在调用时必须用 key=value 的形式来调用,这类参数我们称之类名称参数,而不需要指定参数名称的参数,也称为位置参数。必输参数和可变参数都可以通过位置参数来匹配。如:
def register(name, email, **kwargs):
print('name:%s, age:%s, others:%s', (name, email, kw))
# 错误调用:
register("123","123","123")
# 它会把第三个参数 "123" 也当成是位置参数传入函数
# 相当于调用了一个有三个必输参数的函数
# 而实际的 register 函数只接受两个必输参数
# 因此会报错:TypeError: register() takes 2 positional arguments but 3 were given
12、最后一个比较特殊,也是一个结论:对于任意函数,都可以通过func_name(*args, **kw)的形式来进行调用其中 args 是一个已经获得的列表或者元组,而 kw 是一个已经获得的字典。列表传入会按顺序对必输参数赋值,多余的参数会当成可变参数传入,而字典传入会根据 key 来匹配所有参数。如:
def test1(a, b, c=0, *args, **kwargs):
print('a =', a, 'b =', b, 'c =', c, 'args =', args, 'kw =', kw)
def test2(a, b, c=0, *, d, **kwargs):
print('a =', a, 'b =', b, 'c =', c, 'd =', d, 'kw =', kw)
# 定义一个元组和字典用作参数传入
args = (1, 2, 3, 4)
kw = {'d': 99, 'x': '#'}
test1(*args, **kw)
# a = 1 b = 2 c = 3 args = (4,) kw = {'d': 99, 'x': '#'}
test2(*args, **kw)
a = 1 b = 2 c = 3 d = 88 kw = {'x': '#'}