本文来源:http://blog.csdn.net/callinglove/article/details/45483097

让我们通过以下6步来理解:

1. 通过一个函数调用来理解’*’的作用
2. 通过一个函数的定义来理解’*args’的含义
3. 通过一个函数的调用来理解’**’的作用
4. 通过一个函数的定义来解’**kwargs’的含义
5.注意点:参数arg、*args、**kwargs三个参数的位置必须是一定的。必须是(arg,*args,**kwargs)这个顺序,否则程序会报错。
6. 通过一个应用实例来说明’args’,’kwargs’应用场景以及为何要使用它

1、通过一个函数调用来理解’*’的作用

def fun(a,b,c):
print(a,b,c)
fun(1,2,3) #1 2 3
l=[1,2,3]
fun(*l) #1 2 3
l=(1,2,3)
fun(*l) #1 2 3 元组也可以
l=[1,2,3,4]
fun(*l) #TypeError: fun() takes 3 positional arguments but 4 were given

‘*’ 做了什么?

它拆开数列’l’的数值作为位置参数,并吧这些位置参数传给函数’fun’来调用。

因此拆数列、传位置参数意味着fun(*l)与fun(1,2,3)是等效的,因为l = [1,2,3]

数列’l’含有四个数值.因此,我们试图调用’fun(*l)’,’l’中数值拆开传给函数fun作为位置参数。

但是,’l’中有四个数值,调用’fun(*l)’相当于调用’fun(3,6,9,1)’,又因为函数’fun’定义中只用三个位置参数,因此我们得到这个错误。

2、通过一个函数的定义来理解’*args’的含义

def fun(*args):
print(args,type(args))
fun(10) #(10,) <class 'tuple'>
fun(10,20) #(10,20) <class 'tuple'>
fun([1,2],20,[2,3]) #([1, 2],) <class 'tuple'>

#代码2
def fun(x,y,*args):
print(x,y,args)
fun(1,2,3,4,5) #1 2 (3, 4, 5)
#代码3
def fun(x,y,*args,z):
print(x,y,args,z)
fun(1,2,3,4,5) #TypeError: fun() missing 1 required keyword-only argument: 'z'

可见:1、*args 用来将参数打包成tuple给函数体调用,参数个数不受限制,*args接收元组作为位置参数。

2、由代码2知:x为1,y为2,即第一个和第二个位置参数,之后只有一个参数*args,因此,*args接收除第一个和第二个参数之外的参数作为元组,即(3,4,5)。

3、由代码3知:位置参数z不能放置在*args之后

3、通过一个函数的调用来理解’**’的作用

def fun(a,b,c):
print(a,b,c)
fun(1,2,3) #1 2 3
fun(a=1,b=2,c=3) #1 2 3
d={'b':2,'c':3}
fun(1,**d) #1 2 3
dict={'b':2,'c':3}
fun(1,**dict) #1 2 3
d={'a':1,'b':2,'c':3,'d':4}
fun(**d) #TypeError: fun() got an unexpected keyword argument 'd'
d={'a':1,'b':2,'d':4}
fun(**d) #TypeError: fun() got an unexpected keyword argument 'd'

在函数调用中使用”*”,我们需要元组;在函数调用中使用”**”,我们需要一个字典,字典中参数个数不能多,也不能少。

4、通过函数定义来理解’**kwargs’的含义

def fun(a,**kwargs):
print(a,kwargs)
fun(1,b=2,c=3) #1 {'b': 2, 'c': 3}
fun(1,**{'b':2,'c':3}) #1 {'b': 2, 'c': 3}
fun(1,{'b':2,'c':3}) #TypeError: fun() takes 1 positional argument but 2 were given

在函数定义中”**kwargs”意味着什么?
用”**kwargs”定义函数,kwargs接收除常规参数列表之外的键值参数字典,参数个数不固定,kwargs是个字典。

可以多传参数吗?因为参数不固定,所以也就没有多少的概念了。

5、注意点:参数arg、*args、**kwargs三个参数的位置必须是一定的。必须是(arg,*args,**kwargs)这个顺序,否则程序会报错。

6、通过一个应用实例来说明’args’,’kwargs’应用场景以及为何要使用它

class Model:
def __init__(self,name):
self.name=name
def save(self,force_update=False,force_insert=False):
if force_update and force_insert:
raise ValueError('cannt perform both operations') #故意写作cannt而非cannot
if force_update:
print('updated an existing record')
if force_insert:
print('created a new record') class Child(Model):
def save(self,*args,**kwargs):
if self.name=='abcd':
super().save(*args,**kwargs)
#super(Model,self).save(*args,**kwargs)
else:
return None
child=Child('abcd')
child.save(force_update=True)
child.save(force_insert=True)
child.save(force_insert=True,force_update=True)
# updated an existing record
# created a new record
# ValueError: cannt perform both operations

实际上对应的保存动作发生在’Model’的’save’方法中。所以我们调用子类的的’save()’方法而非’Model’的方法.子类Child的’save()’接收任何父类save()需要的参数,并传给父类方法。因此,子类’save()’方法参数列表中有”*args”和”**kwargs”,它们可以接收任意位置参数或键值参数,常规参数列表除外

理解python中的'*','*args','**','**kwargs'的更多相关文章

  1. 理解理解python中的'*','*args','**','**kwargs'

    http://blog.csdn.net/callinglove/article/details/45483097 讲了一大堆, 我也是用来理解类继承当中的参数行为的. =============== ...

  2. 理解 Python 中的 *args 和 **kwargs

    Python是支持可变参数的,最简单的方法莫过于使用默认参数,例如: def test_defargs(one, two = 2): print 'Required argument: ', one ...

  3. python 中参数*args, **kwargs

    def func(*args, **kwargs): print 'args : ', args print 'kwargs :', kwargs def test(): func(1,2,'x',' ...

  4. 理解 Python 中的可变参数 *args 和 **kwargs:

    默认参数:  Python是支持可变参数的,最简单的方法莫过于使用默认参数,例如: def getSum(x,y=5): print "x:", x print "y:& ...

  5. [转]深刻理解Python中的元类(metaclass)以及元类实现单例模式

    使用元类 深刻理解Python中的元类(metaclass)以及元类实现单例模式 在看一些框架源代码的过程中碰到很多元类的实例,看起来很吃力很晦涩:在看python cookbook中关于元类创建单例 ...

  6. python可变参数*args, **kwargs

    python可变参数*args, **kwargs def foo(* args, ** kwargs): print ' args = ',  args print ' kwargs = ',  k ...

  7. 【转】你真的理解Python中MRO算法吗?

    你真的理解Python中MRO算法吗? MRO(Method Resolution Order):方法解析顺序. Python语言包含了很多优秀的特性,其中多重继承就是其中之一,但是多重继承会引发很多 ...

  8. 深入理解Python中的yield和send

    send方法和next方法唯一的区别是在执行send方法会首先把上一次挂起的yield语句的返回值通过参数设定,从而实现与生成器方法的交互. 但是需要注意,在一个生成器对象没有执行next方法之前,由 ...

  9. 如何理解python中的if __name__=='main'的作用

    一. 一个浅显易懂的比喻 我们在学习python编程时,不可避免的会遇到if __name__=='main'这样的语句,它到底有什么作用呢? <如何简单地理解Python中的if __name ...

随机推荐

  1. dMd----攻防世界

    首先在Linux上查看题目,没有什么发现elf文件,之后使用ida打开看看,找到main函数,f5查看, 上图一些字符是char过的,便于查看,发现是一个if else语句,先经过了MD5加密然后判断 ...

  2. webpack 4 x使用详细

    1.首先安装node.js 2.打开控制台cmd,输入npm install webpack webpack-cli webpack-dev-server -g 3.在本地磁盘上建一个文件夹,然后通过 ...

  3. Koa2+mongoose

    为什么选择Koa koa是Express框架同个公司的产品,是开发者在node7.0版本之后使用promise的api把express再次封装了一次,起名Koa,==Koa=Express+Promi ...

  4. 向量容器vector操作

    1.向量容器vector 1.1 vector说明 进行vector操作前应添加头文件#include<vector>: vector是向量类型,可以容纳许多类型的数据,因此也被称为容器: ...

  5. 001.CI4框架CodeIgniter的默认访问路径url

    1. 我们解压缩CI4的压缩包,找到app目录,点开Controllers目录,在Home.php文件中,写入我们的如下代码: 002.我们来访问我们的网站 http://127.0.0.1/CI4/ ...

  6. 007-PHP变量和函数相互转换

    <?php function write($text) //定义function write()函数 { print($text); //打印字符串 } function writeBold($ ...

  7. Linux学习《第四章shell脚本练习一》随堂练习(重要)

  8. Linux下的文件目录树结构

    Linux下的文件目录及文件结构 一.文件和文件夹 在Linux系统下,一切皆是文件.就连Linux本身也是基于文件表示的操作系统. 1.文件:文件在Linux系统之下,一般分为两种:一是一般性文件, ...

  9. bool之regexp正则注入(原理详解)

    感谢原创博主的文章,在此致敬.本文转自:http://www.cnblogs.com/lcamry/articles/5717442.html 我们都已经知道,在MYSQL 5+中 informati ...

  10. 吴裕雄 Bootstrap 前端框架开发——Bootstrap 字体图标(Glyphicons):glyphicon glyphicon-font

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name ...