前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >每日一题:如何理解不定长参数

每日一题:如何理解不定长参数

作者头像
用户7685359
发布2020-08-24 16:20:28
6330
发布2020-08-24 16:20:28
举报
文章被收录于专栏:FluentStudyFluentStudy

点击关注了解更多精彩内容!!

题目描述

题目描述:

1、怎么理解不定长参数?

2、*args 和 **kwargs 是什么意思?为什么要使用它们?

答案要点如下:

1、函数参数可为分如下几种:必选参数、默认参数、可变参数、命名关键字参数和关键字参数

2、当我们在定义和调用一个函数时,如果包含所有的参数类型,则必须按照:必选参数、默认参数、可变参数、命名关键字参数和关键字参数的顺序。但是在实际开发中,不建议包含过多的参数类型,会影响代码的可读性

3、必选参数很简单,就是函数中必须要接受的参数

4、默认参数,即给参数赋一个默认值,我们在传递时,可省略对该参数的传值操作。如:

代码语言:javascript
复制
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 的方法来操作参数,示例如下:

代码语言:javascript
复制
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语言 中的指针),示例如下:

代码语言:javascript
复制
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 传入函数。它用于扩展函数的功能。比如我们要实现用户注册,有必输项和非必输项,这些非必输项就可以用关键字参数来接受。示例如下:

代码语言:javascript
复制
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示例如下:

代码语言:javascript
复制
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、命名关键字参数用于限制调用函数可传入的属性。这里的限制是假限制,因为仍然可以传入,只是函数体中对限制外的参数不作任何处理。命名关键字参数用一个 * 号分隔,* 后面的参数都被视为命名关键字参数。如:

代码语言:javascript
复制
def person(name, age, *, city, job):
    print(name, age, city, job)

10、如果函数中已经有了一个可变参数的定义,后面的命名关键字参数则不需要再添加 * ,如:

代码语言:javascript
复制
def person(name, age, *args, city, job):
    print(name, age, args, city, job)

11、关键字参数和命名关键字参数在调用时必须用 key=value 的形式来调用,这类参数我们称之类名称参数,而不需要指定参数名称的参数,也称为位置参数。必输参数和可变参数都可以通过位置参数来匹配。如:

代码语言:javascript
复制
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 来匹配所有参数。如:

代码语言:javascript
复制
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': '#'}

你点的每个赞,我都认真当成了喜欢

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2019-06-25,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 FluentStudy 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档