Python函数参数传递机制

最近在写代码的过程中,发现Python参数传递不是很明白。Python确实很灵活,但是灵活的后果就是要花更多的时间去研究。废话不多说,始めましょう!!!

Python参数传递有一下几种方式:

1:位置参数

Fun(arg1,arg2,...)

2:默认值参数

Fun(arg1,arg2=<value>...)

3:关键字参数

Fun(arg1=<value>,arg2=<value>...)

4:过量位置参数

Fun(arg1,arg2,*arg3)

5:过量关键字参数

Fun(arg1,arg2,**arg3)

6:混合参数

Fun(arg1,arg2=<value>,*arg3,**arg4)

一:位置参数

Fun(arg1,arg2,...)

位置参数在所有的开发语言中基本都有,这种方式函数声明了几个参数,在调用的时候就必须传几个参数,并且传递参数的位置也要跟声明函数的参数位置必须一致。

In [64]: def demo(a,b):
   ....:     return a+b
   ....: 

In [65]: demo(1)
---------------------------------------------------------------------------
TypeError                                 Traceback (most recent call last)
<ipython-input-65-cd1a36a59e96> in <module>()
----> 1 demo(1)

TypeError: demo() takes exactly 2 arguments (1 given)

In [66]: demo(1,2)
Out[66]: 3

In [67]:

这种方式比较简单,就不做太多说明。

二:默认值传递

Fun(arg1,arg2=<value>...)

我个人认为,默认值传递和第三种的关键字传递结合在一起说明比较好,因为大多数情况下,默认值传递本身就要使用关键字。

In [67]: def demo(a,b=2):
   ....:     return a+b
   ....: 

In [68]: demo(2)
Out[68]: 4

In [69]: demo(4,5)
Out[69]: 9

In [70]: demo(a=4)
Out[70]: 6

In [71]: demo(a=4,b=5)
Out[71]: 9

In [72]: demo(b=5)
---------------------------------------------------------------------------
TypeError                                 Traceback (most recent call last)
<ipython-input-72-f88a17b9de38> in <module>()
----> 1 demo(b=5)

TypeError: demo() takes at least 1 argument (1 given)

In [73]: demo(b=5,a=3)
Out[73]: 8

在声明函数demo的时候,给参数b付了一个默认值2,所以在调用demo的时候,可以只给a传递,在函数内部计算的时候就使用的是b的默认值。

如果在调用的时候给b传了一个值,那就使用传递的值,而不是使用默认值。

但是这种情况下,参数a是必须的,所以在调用的时候必须给a传值,否则将报错。因为参数b有默认值,所以即使不给b传值,也是可以的,将使用默认值。

在声明函数的时候,参数列表中有参数的名称,所以可以通过指定变量名的形式传递参数,并且变量的位置可以改变,这样Python内部自己就会去对应哪个实参该传给哪个形参。

三:关键字传递

Fun(arg1=<value>,arg2=<value>...)

In [79]: def demo(a=1,b=2,c=3):
   ....:     return a+b+c
   ....: 

In [81]: demo(4,5,6)
Out[81]: 15

In [82]: demo(4)
Out[82]: 9

In [83]: demo(b=4)
Out[83]: 8

In [84]: demo(b=8,a=5,c=3)
Out[84]: 16

理解了上面默认值传递的形式,再看关键字传递就很容易理解了,所以就不在做太多的解释。

四:过量位置参数

Fun(arg1,arg2,*arg3)

In [94]: def demo(a,b,*args):
   ....:     print a+b
   ....:     print type(args)
   ....:     print args
   ....:     

In [95]: demo(1,2,3,4,5)
3
<type ‘tuple‘>
(3, 4, 5)

In [100]: demo(1,2,3)
3
<type ‘tuple‘>
(3,)

In [98]: demo(1,23)
24
<type ‘tuple‘>
()

In [101]: demo(1)
---------------------------------------------------------------------------
TypeError                                 Traceback (most recent call last)
<ipython-input-101-cd1a36a59e96> in <module>()
----> 1 demo(1)

TypeError: demo() takes at least 2 arguments (1 given)

如上面的代码所示:声明的函数demo中,参数a和b是必须参数,所以a和b是必须的,也就是说在调用demo函数的时候,至少需要传递两个参数,否则将出错。

过量位置参数使用星号(*)加变量名的形式(*args),在传递多余2个参数的时候,前两个参数分别赋值给了a和b,后面的参数都将整体赋值给args,通过打印可以看出,args是一个元祖类型,所以可以通过遍历将里面的数据取出来使用。

五:过量关键字参数

Fun(arg1,arg2,**arg3)

In [103]: def demo(a,b,**kwargs):
   .....:     print a+b
   .....:     print type(kwargs)
   .....:     print kwargs
   .....:     

In [104]: demo(1,2)
3
<type ‘dict‘>
{}

In [105]: demo(1,2,c=1)
3
<type ‘dict‘>
{‘c‘: 1}

In [106]: demo(1,2,c=1,d=2)
3
<type ‘dict‘>
{‘c‘: 1, ‘d‘: 2}

In [107]: demo(1,2,3,c=1,d=2)
---------------------------------------------------------------------------
TypeError                                 Traceback (most recent call last)
<ipython-input-107-94e09001d957> in <module>()
----> 1 demo(1,2,3,c=1,d=2)

TypeError: demo() takes exactly 2 arguments (5 given)

In [108]: demo(1,2,3)
---------------------------------------------------------------------------
TypeError                                 Traceback (most recent call last)
<ipython-input-108-856c09b109ab> in <module>()
----> 1 demo(1,2,3)

TypeError: demo() takes exactly 2 arguments (3 given)

过量关键字使用两个星号加变量名的形式,即(**kwargs)

在之前的介绍关键字参数的时候可以看出,关键字参数一般都以键值对的形式存在,在过量关键字参数中也是,kwargs只接受键值对形式的参数,所以像107和108这两种调用方式就会出错,值1和2分别赋值给了a和b,后面的c=1和d=2通过kwargs来接受,剩余的3没有形式参数来接受,所以就会报错。因为**kwargs值接受键值对形式的参数。

六:混合参数

Fun(arg1,arg2=<value>,*arg3,**arg4)

In [109]: def demo(a,b,c=1,*args,**kwargs):
   .....:     print a+b+c
   .....:     print args
   .....:     print kwargs
   .....:     

In [110]: demo(2,3)
6
()
{}

In [111]: demo(2,3,4)
9
()
{}

In [112]: demo(2,3,4,5)
9
(5,)
{}

In [113]: demo(2,3,4,5,6)
9
(5, 6)
{}

In [114]: demo(2,3,4,5,6,d=1,e=2)
9
(5, 6)
{‘e‘: 2, ‘d‘: 1}

如上面代码所示,混合参数就是将前面介绍的几种形式结合起来一起使用。

a和b是位置参数,c是默认值参数,args是过量位置参数,kwargs是过量关键字参数。

理解了上面说的几种形式,再来理解混合参数形式应该比较简单,就不做过多的解释。

下面通过混合实例的形式,说明一下具体的使用。

首先定义函数如下

In [109]: def demo(a,b,c=1,*args,**kwargs):
   .....:     print a+b+c
   .....:     print args
   .....:     print kwargs
   .....:

实例1:

In [115]: demo(2)
---------------------------------------------------------------------------
TypeError                                 Traceback (most recent call last)
<ipython-input-115-5fc919890c38> in <module>()
----> 1 demo(2)

TypeError: demo() takes at least 2 arguments (1 given)

在调用函数的时候只传了一个值,但是从函数定义来看,a和b时必须的,只传了一个值,相当于只给a赋值了,b没有赋值,所以会报错。

实例2

In [117]: demo(2,3,a=4)
---------------------------------------------------------------------------
TypeError                                 Traceback (most recent call last)
<ipython-input-117-803c2126718d> in <module>()
----> 1 demo(2,3,a=4)

TypeError: demo() got multiple values for keyword argument ‘a‘

上面调用函数出错的原因是因为没有按照正确的顺序给demo赋值,不能就认为指定a=4,就会在调用的时候就将4赋值给了a,剩下两个值2和3分别赋值给b和c,这是错误的,在这种混合模式下,赋值顺序必须是先位置参数->默认值参数->过量位置参数->过量关键字参数,否则就会出错。再看下面这个例子。

In [119]: demo(1,2,3,d=2,5,6)
  File "<ipython-input-119-6a415509517c>", line 1
    demo(1,2,3,d=2,5,6)
SyntaxError: non-keyword arg after keyword arg

In [120]: demo(1,2,3,5,6,d=2,)
6
(5, 6)
{‘d‘: 2}

1,2,3分别赋值给了a,b,c,这没有问题,但是将d=2过量关键字参数放在了5,6过量位置参数前面,就出现了报错,将两者的顺序调换一下就没有问题。

重新定义函数demo,之后的实例将使用新定义的函数

In [121]: def demo(a,b,c):
   .....:     return a+b+c
   .....:

实例3

In [122]: demo(a=1,2,3)
  File "<ipython-input-122-ec666e77af79>", line 1
    demo(a=1,2,3)
SyntaxError: non-keyword arg after keyword arg

In [123]: demo(1,2,c=3)
Out[123]: 6

In [124]: demo(a=1,b=2,c=3)
Out[124]: 6

如上面的代码所示:

[122]在调用函数demo的时候在最前面指定了a的值,出现了错误,错误信息翻译成中文意思是:不要在关键字参数后使用位置参数。

[123]的调用方式就不会出现错误。所以在调用函数的时候,使用关键字形式传参一定要在位置传参的后面。

实例4

In [125]: args = (2,3)

In [126]: demo(1,*args)
Out[126]: 6

In [127]: demo(a=1,*args)
---------------------------------------------------------------------------
TypeError                                 Traceback (most recent call last)
<ipython-input-127-aeb489dc72f3> in <module>()
----> 1 demo(a=1,*args)

TypeError: demo() got multiple values for keyword argument ‘a‘

上面[126]调用的时候,通过一个元祖包装了2个参数值,1赋值给形参a,元祖args中的2,3分别赋值给了形参b和c。

但是[127]的调用形式就会出错,其调用形式可以看成demo(a=1,2,3),原因就跟上面实例3的情况差不多,因为Python中“关键字形式传参一定要在位置传参的后面”,所以Python在赋值的时候,将先将2赋值给a,将3赋值给b,最后又将a=1赋值给a,所以就会出现上述的错误信息。

如果换一种形式传参,将2和3分装到字典中,就不会出现这种错误,如下代码所示

In [129]: dict={‘b‘:2,‘c‘:3}

In [130]: demo(a=1,**dict)
Out[130]: 6

In [131]: demo(**dict,a=1)
  File "<ipython-input-131-cd33424dd41b>", line 1
    demo(**dict,a=1)
               ^
SyntaxError: invalid syntax

将参数包装到字典中,就相当于所有的参数都是用关键字参数的形式传递,即demo(a=1,b=2,c=3),所以就不会出错。

综上所述,在传递参数的时候,一定要遵循参数传递的顺序,即位置参数->默认值参数->过量位置参数->过量关键字参数。

时间: 2024-10-10 11:08:22

Python函数参数传递机制的相关文章

C/C++中的函数参数传递机制

对函数的形参感兴趣的可以看一下 一. 函数参数传递机制的基本理论 函数参数传递机制问题在本质上是调用函数(过程)和被调用函数(过程)在调用发生时进行通信的方法问题.基本的参数传递机制有两种:值传递和引用传递.以下讨论称调用其他函数的函数为主调函数,被调用的函数为被调函数. 值传递(passl-by-value)过程中,被调函数的形式参数作为被调函数的局部变量处理,即在堆栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实参的一个副本.值传递的特点是被调函数对形式参数的任何操作都是作为

Python核心技术与实战——十三|Python中参数传递机制

我们在前面的章节里学习了Python的函数基础以及应用,那么现在想一想:传参,也就是把一些参数从一个函数传递到另一个函数,从而使其执行相应的任务,这个过程的底层是如何工作的,原理又是怎样的呢? 在实际过程中,我们写完了代码测试时候发现结果和预期值不一样,在一次次debug后发现是传参过程中数据结构发生了改变,导致程序出错.比富我们把一个列表作为实参传递给另一个函数,但是我们并不希望列表再函数运行结束后发生变化.但往往事与愿违,由于某些额外的操作改变了他的值,那就导致后续程序一系列错误的发生.因此

python函数参数传递总结

默认参数 def enroll(name, gender, age=6, city='Beijing'):    print('name:', name)    print('gender:', gender)    print('age:', age)    print('city:', city) 有多个默认参数时,调用的时候,既可以按顺序提供默认参数,比如调用enroll('Bob', 'M', 7),意思是,除了name,gender这两个参数外,最后1个参数应用在参数age上,city

Python 函数 参数传递

参数传递    在 python 中,类型属于对象,变量是没有类型的:        a=[1,2,3]        a="Runoob"    以上代码中,[1,2,3] 是 List 类型,"Runoob" 是 String 类型,而变量 a 是没有类型,她仅仅是一个对象的引用(一个指针),可以是指向 List 类型对象,也可以是指向 String 类型对象. 可更改(mutable)与不可更改(immutable)对象    在 python 中,strin

python 函数参数传递

参数传递: 参数的传递是通过自动将对象赋值给本地变量来实现的,作为参数被传递的对象从不自动拷贝 在函数内部的参数名的赋值不会影响调用者 改变函数的可变对象参数的值也许会对调用者有影响(实质:不可变对象(如字符串 简单类型 元组)是通过'值'进行传递的,可变对象如字典 列表是通过'指针'进行传递的) >>> def change(a,b): a=2 print id(b) print b b[0]='spam' print b >>> x=1 >>> l

[蟒蛇菜谱]Python函数参数传递最佳实践

将函数作为参数传递,同时将该函数需要的参数一起传递.可参考threading.Timer的处理方式: class threading.Timer(interval, function, args=[], kwargs={}) Create a timer that will run function with arguments args and keyword arguments kwargs, after interval seconds have passed. 内部实现中,调用funct

python中的*和**参数传递机制

python的参数传递机制具有值传递(int.float等值数据类型)和引用传递(以字典.列表等非值对象数据类型为代表)两种基本机制以及方便的关键字传递特性(直接使用函数的形参名指定实参的传递目标,如函数定义为def f(a,b,c),那么在调用时可以采用f(b=1,c=2,a=3)的指定形参目标的传递方式,而不必拘泥于c语言之类的形参和实参按位置对应) 除此之外,python中还允许包裹方式的参数传递,这未不确定参数个数和参数类型的函数调用提供了基础: def f(*a,**b) 包裹参数传递

Python的函数参数传递

首先还是应该科普下函数参数传递机制,传值和传引用是什么意思? 函数参数传递机制问题在本质上是调用函数(过程)和被调用函数(过程)在调用发生时进行通信的方法问题.基本的参数传递机制有两种:值传递和引用传递. 值传递(passl-by-value)过程中,被调函数的形式参数作为被调函数的局部变量处理,即在堆栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实参的一个副本.值传递的特点是被调函数对形式参数的任何操作都是作为局部变量进行,不会影响主调函数的实参变量的值. 引用传递(pass-

深入剖析C/C++函数的参数传递机制

2014-07-29 20:16 深入剖析C/C++函数的参数传递机制 C语言的函数入口参数,可以使用值传递和指针传递方式,C++又多了引用(reference)传递方式.引用传递方式在使用上类似于值传递,而其传递的性质又象是指针传递,这是C++初学者经常感到困惑的.为深入介绍这三种参数传递方式,我们先把话题扯远些: 1. C/C++函数调用机制及值传递: 在结构化程序设计方法中,先辈们告诉我们,采用“自顶向下,逐步细化”的方法将一个现实的复杂问题分成多个简单的问题来解决.而细化到了最底层,就是