Python中args用法,python **kwargs用法
本文主要介绍了python中*args和**kwarsg、闭包和decorators的使用说明,有很好的参考价值。希望对大家有帮助。如有错误或不足之处,请不吝赐教。
目录
*args和**kwarsg,以及* *args,**kwargs在Python fun中(*args,**kwargs)。1.python函数中的两个参数。2.理解*3。懂**4。理解函数调用中的*args。
*args与**kwarsg及闭包和装饰器
过程
首先理解闭包,然后是装饰器,不要忘记可变长度参数。
定义函数():
msg=111
定义函数1():
打印(邮件)
返回功能1
1-理解闭包
闭包意味着内部函数调用外部函数范围内的变量。
例如,func1是一个闭包函数。
Func()()#这其实是func1()
2-装饰者
Fn是修饰的目标函数。
2.1-它只是一个用于传递函数名的装饰器[基本不使用]
2.2-用参数修饰函数
2.3-用返回值修饰函数
2.4-具有不确定装饰参数的功能[可归类为具有装饰参数的功能]
2.5-装饰器本身携带的参数
定义装饰者(fn):
定义包装():
打印(“添加的函数,不带参数的装饰函数”)
返回fn()
返回包装
@装饰者
定义测试():
打印(“原始功能”)
Test()#实际上是装饰器(Test)
def装饰工1(fn):
def包装(n1,n2):
打印(“添加的函数,用参数修饰函数”)
返回fn(n1,n2)
返回包装
@decorator1
定义测试1(a,b):
打印( a b=%s%(a b))
打印(“原始功能”)
Test1(1,2)#实际上是decorator1(test1(1,2))
def解码器2(fn):
定义包装():
打印(“添加的函数,用返回值修饰函数”)
res=fn()
返回资源
返回包装
@decoretor2
定义测试2():
打印(“原始功能”)
返回返回值001
A=test2() #实际上是decorator2(test2)
打印(一份)
def装饰工3(fn):
def warpper(*args,**kwargs):
Print(“添加的函数,用可变长度参数修饰的函数”)
返回fn(*args,**kwargs)
返回warpper
@decorator3
def测试3(n1、n2、n3):
打印(“原始功能”)
打印(n1 n2 n3)
Test3(1,2,3)#实际上是decorator1(test1(1,2,3))
def装饰工4(家庭):
定义函数_1(fn):
def包装(*args,**kwargs):
Print(装饰器本身带有参数)
打印(当前主目录位于%s%(主目录))
返回fn(*args,**kwargs)
返回包装
返回func_1
@decorator4(home=武汉)
定义测试4(n1、n2、n3):
打印(“原始功能”)
打印(n1 n2 n3)
# test3(1,2,3)=decorator3(home="武汉")(test(1,2,3))()
"""
1-先调用decorator3(home="wuhan")
2-执行func_1(test(1,2,3)) # 到这里其实就和前面的装饰器一样
3-执行wrapper
4-执行test(1,2,3)
"""
test4(1,2,3)
Python fun(*args,**kwargs)中*args,**kwargs参数含义及用法
1. Python函数中的两种参数
我们知道,在Python中有两种参数
- 位置参数(positional argument): 位置参数只能由参数位置决定
- 关键词参数(keyword argument): 关键词参数只需要用 keyword = somekey 的方法即可传参
位置参数只能由参数位置决定。这也就决定了位置参数一定要在前面,否则关键词参数数量的变化都会使得位置无法判断。
2. 理解函数调用中的*
*的作用是将tuple或者list中的元素进行unpack,分开传入,作为多个参数。
def func(a,b,c)print(a,b,c)
alist = [1,2,3] # 这里alist的长度必须和函数中参数的个数相同,否则会报错
func(*alist) # 等同于 func(1, 2, 3)
1 2 3
2.1 * 做了什么
它拆开数列alist的数值作为位置参数,并把这些位置参数传给函数func来调用。
因此拆数列、传位置参数意味着func(*alist)与func(1,2,3)是等效的,因为 alist= [1,2,3]。
3. 理解函数调用中的**
**的作用是unpack字典,并将字典中的数据项作为键值参数传给函数。
为了更好的理解举几个例子:
def func(a, b, c):print(a, b, c)
if __name__ == "__main__":
dic = {b: 2, c: 3}
func(1, b=2, c=3)
func(1, **dic)
1 2 3
1 2 3
4. 理解函数调用中的*args和**kwargs
kwargs是keyword argument的缩写,args就是argument。常见的是*args 在 **kwargs 前面。
这两个的用途和效果如下:
def this_fun(a,b,*args,**kwargs):"""
在这个函数定义中,参数”a, b”代表”常规参数列表”。
args 接收元组作为位置参数,而非是常见的参数列表
"""
print(a,b)
print(args)
print(kwargs)
if __name__ = __main__
this_fun(0,1,2,3,index1=11,index2=22)
0,1
(2, 3)
{index2: 22, index1: 11}
也就是说,第一中不定的参数形式把剩下的没有关键字的参数收起来形成一个tuple,而第二种把有关键字的收起来做成一个字典。
5. 实例说明args, kwargs的应用场景
5.1 子类传参给父类方法
在任何时候继承类和重写方法的,我们应当用到args, kwargs将接收到的位置参数和键值参数给父类方法。通过实例我们更好的理解
class Model(object):def __init__(self, name):
self.name = name
def save(self, force_update=False, force_insert=False):
if force_update and force_insert:
raise ValueError("Cannot perform both operations")
if force_update:
print("Updated an existing record")
if force_insert:
print("Created a new record")
定义一个类,我们可以创建类的对象,类的对象有一个方法save().假设类的对象可以通过save()方法保存到数据库中。通过函数save()参数来决定是否在数据库中创建一条记录或者更新现存的记录。
构造一个新类,类有Model的行为,但只有符合某些条件才会保存这个类的对象。这个新类继承Model,重写Model的save()
class ChildModel(Model):def save(self, *args, **kwargs):
if self.name == abcd:
super(ChildModel, self).save(*args, **kwargs)
else:
return None
实际上对应的保存动作发生在’Model’的save方法中。所以我们调用子类的的save()方法而非’Model’的方法.子类ChildModel的save()接收任何父类save()需要的参数,并传给父类方法。因此,子类save()方法参数列表中有*args和**kwargs,它们可以接收任意位置参数或键值参数,常规参数列表除外。
下面创建ChildModel实体并调用save方法:
c=ChildModel(abcd)c.save(force_insert=True)
c.save(force_update=True)
# 结果
Created a new record
Updated an existing record
这里传参数给对象的save()方法。调用的是子类的save(), 它接收一个包含关键字参数kwargs的字典。然后,它使用**将字典作为关键字参数unpack,然后将其传递给超类save()。因此,超类save()获得关键字参数force_insert并执行相应的操作。
5.2 *args 实现sum
def my_sum(*args):res = 0
for val in args:
res += val
return res
l1 = [4, 8]
l2 = [1,2,3]
print(my_sum(*l1)) # 12
print(my_sum(*l2)) # 6
print(my_sum(4,5,6)) # 15
以上为个人经验,希望能给大家一个参考,也希望大家多多支持盛行IT软件开发工作室。
郑重声明:本文由网友发布,不代表盛行IT的观点,版权归原作者所有,仅为传播更多信息之目的,如有侵权请联系,我们将第一时间修改或删除,多谢。