Python 学习 第八篇:函数2(参数、lamdba和函数属性)

函数的参数是参数暴露给外部的接口,向函数传递参数,可以控制函数的流程,函数可以0个、1个或多个参数;在Python中向函数传参,使用的是赋值方式。

一,传递参数

参数是通过赋值来传递的,传递参数的特点是:

  • 参数的传递是通过自动把对象赋值给函数的本地变量名来实现的,
  • 在函数内部的变量名的赋值不会影响调用者,在函数运行时,在函数头部的参数名是一个新的、本地的变量名,这个变量名的作用域是在函数的本地作用域内。
  • 改变函数的可变对象参数的值,会修改原始对象的值。

当通过赋值方式,把可变对象的引用传递给函数时,在函数内部对参数进行修改,在函数外部可以看到修改。

例如,列表是一个可变对象,当把列表 a 传递给 changer函数时,函数向列表末尾添加一个字符a,当函数调用结束时,列表末尾被修改。

>>> def changer(x):
...     x.append('a')
...
>>> a=[1,2,3]
>>> changer(a)
>>> a
[1, 2, 3, 'a']

这是由于可变对象是通过指针进行传递的,当把列表的引用传递给参数时,函数直接在原处修改对象。

也可以通过向参数传递拷贝的方式,使得函数无法修改原始的对象。

changer(a[:])
changer(a.copy())

二,参数匹配模型

不管使用何种参数的匹配模型,参数总是通过赋值进行传递的。

在默认情况下,参数是通过其位置进行匹配的,从左向右,而且必须精确地传递和函数头部参数名一样多的参数。还能够通过参数名匹配,默认参数名,以及对于额外参数的容器来进行传参。

1,位置匹配

一般情况下,参数是通过位置从左至右进行匹配,把参数传递给函数头部的参数名称。

例如,定义函数func,包含三个参数,那么必须使用三个参数传值进行调用:

def func(a,b,c):

调用的格式是,按照顺序把值写在括号中,Python按照位置匹配参数,把1赋值给a,把2赋值给b,把3赋值给c:

func(1,2,3)

2,关键字匹配

调用者使用arg_name=value的方式,通过在调用时使用参数的变量名进行匹配。

例如,定义参数func,包含三个参数,那么必须使用三个参数传值进行调用:

def func(x,y,z):

使用关键字调用的格式是:按照 arg=named的方式进行匹配,可以不按照位置:

func(y=2, x=1, z=3)

在调用函数时,Python按照参数名进行匹配,把2传递给参数y,把1传递给参数x,把3传递给参数z。

也可以混合使用位置匹配和关键字匹配,例如,下面的函数调用:

func(1,z=3,y=2)

基于位置的匹配参数,首先按照从左至右的顺序匹配前面的参数,之后,再按照基于变量名进行关键字匹配。

3,默认参数

在定义函数时,可以为函数的参数赋值,这就是参数的默认值。当没有为该参数传值时,参数值使用默认值。

例如,定义一个函数,包含三个参数,x、y和z,其中参数z有默认值1:

def func(x,y,z=1):

当调用这个函数,必须为参数x和y传值,无论是通过位置匹配还是关键字匹配来实现都可以。然而,参数z的传值是可选的。如果不为z传值,那么z使用默认值;如果为z传值,那么z使用传递的值。

func(2,3)

4,可变长参数列表

可变参数用于收集任意多个基于位置或关键字的参数,在函数定义时,如果参数名前带一个*号,那么该参数收集的是基于位置的传值;如果参数名前带两个*号(**arg),那么该参数收集的是基于关键字(arg=value)的传值。

def func(*args)
def func(**dict)

这两种方式的不同点是,*args方式是把任意多个传值封装成元组(arg1,arg2),而**dict是把任意多个 arg=value 封装成字典{arg:value},字典的key是参数名,字典key对应的值是参数的值。

例如,定义函数foo,包含一个位置匹配参数,1个可变的位置匹配参数,1个可变的关键字匹配参数:

def foo(a,*b,**c):

在调用函数时,首先按照位置匹配,把1传递给参数a,把元组(2,3)传递给参数b,把字典{x:1,y:2}传递给参数c:

foo(1,2,3,x=1,y=2)

5,只能使用关键字参数

参数必须按照名称传值,在函数定义时,参数必须在调用中按照关键字进行传值:

def func(*,arg=value)

从语法上来讲,keyword-only参数编码为关键字参数,出现在参数列表的*args之后,所有这些参数都必须在调用中使用关键字语法来传值。

def foo(a,*b,c):

在调用foo函数时,参数a可以按照位置和关键字来传值,参数b接收按照其余的按照位置来传值的变量,参数c必须按照关键字来传值:

foo(1,2,'a',3,c='key')

也可以在参数列表中使用一个*字符,来表示一个函数不会接受一个可变的参数列表,而是仍然期待跟在*后面的所有参数都作为关键字匹配。

例如,定义函数foo,只接受3个参数,不接受可变长度的参数列表:

def foo(a,*,b,c):

在调用时,参数a可以按照位置和关键字来传值,参数b和c必须按照关键字来传值:

foo(1,b='x',c=2)

三,lambda表达式

除了def语句之外,Python还提供了一种生成函数对象的表达式形式,lambda表达式。就像def语句一样,这个表达式创建了一个函数对象,但是它返回了一个函数,而不是把这个函数赋值给一个变量名。

lambda表达式的一般形式是关键字lambda,之后是一个或多个参数(与一个def头部内用括号括起来的参数列表极其类似),紧跟一个冒号,之后是一个表达式:

lambda arg1,arg2,... argN: expression_using_args

lambda表达式的特性是:

  • lambda表达式返回的是函数对象;
  • lambda是一个表达式,而不是一个语句,作为一个表达式,lambda返回了一个值(一个新的函数),可以选择性地赋值给一个变量名;
  • lambda的主体是一个单个的表达式,而不是一个代码块,这个表达式的结果作为函数的结果返回。

lambda只有一行表达式,只能用于创建简单的函数,而def 可以包含多条语句,用于处理复杂的任务。

对于简单的处理流程,既可以使用def,也可以用lambda来实现,例如:

def func(x,y,z): return x+y+z

func=lambda x,y,z: x+y+z

由于lambda表达式返回的是一个函数对象,因此,可以用于map()、filter()和reduce()函数中:

list(map((lambda x: x+3), range(0,5)))
list(filter((lambda x: x>2), range(0,5)))

from functools import reduce
reduce((lambda x,y: x+y),range(0,5))

reduce()函数对每对元素都应用函数,并运行到最后结果,上述reduce()函数的作用是把序列中的所有元素相加,等价的写法是:

x=list(range(0,5)]
res=x[0]
for i in x[1:] :
    res+=i

四,函数的属性

可以向函数添加任意多个用户定义的属性,函数属性的格式是:函数名称.变量名 

 例如,向函数func中添加state属性:

def func():

func.state='begin'

 

参考文档:

posted @ 2018-12-19 09:26  悦光阴  阅读(1261)  评论(0编辑  收藏  举报