一、函数的定义与创建
函数是 Python 为了代码最大程度的重用和最小化代码冗余而提供的基本程序结构;它能让程序员将
复杂的系统分解为可管理的部件。在Python中可以创建四种函数:
全局函数:定义在模版中 局部函数:嵌套在其他函数中 Lambda函数:又称匿名函数,表达式 方法:与特定数据类型关联的函数,并且只能与数据类型关联一起使用 |
-
定义函数语法
def functionName(parameters):
suite
return value
def是一个可执行语句,因此可以出现在任何能够使用语句的地方,甚至可以嵌套于其他语句,例如 if 或 while;def 创建了一个对象并将其赋值给一个变量名(即函数名);return用于返回结果对象,其为可选;无returne语句的函数自动返回none对象;def 语句运行之后,可以在程序中通过函数后附加括号进行调用。
In [2]: def make_a_sound():
...: print('quack')
...:
In [3]: make_a_sound()
quack
In [4]: a=make_a_sound()
quack
In [5]: type(a)
Out[5]: NoneType
上面是一个最简单的无参数函数,对其调用后python会执行函数内部代码。此函数会打印输出一个单词,由于没有设置返回值,因此默认为NoneType。
-
参数传递形式
当定义的函数有参数时,参数传递有两种形式。
位置参数:从左向右,传入的参数值按顺序一次复制进去 关键字参数:调用参数时按关键名称匹配,无需关心函数定义时参数的位置 |
In [6]: def menu(wine,entree,dessert):
...: return{'wine':wine,'entree':entree,'dessert':dessert}
...:
In [7]: menu('chardonnay','chicken','cake') #依次传递参数,位置参数
Out[7]: {'dessert': 'cake', 'entree': 'chicken', 'wine': 'chardonnay'}
In [8]: menu(entree='beef',dessert='bagel',wine='bordeaux') #按关键字传递参数
Out[8]: {'dessert': 'bagel', 'entree': 'beef', 'wine': 'bordeaux'}
定义函数时可以为参数设置默认的参数值,当调用函数没有指定参数值时就会使用此默认值。定义时无默认值的参数在左边,有默认值的参数在右边。
In [9]: def menu(wine,entree,dessert='pudding'):
...: return{'wine':wine,'entree':entree,'dessert':dessert}
...:
In [10]: menu('chardonnay','chicken')
Out[10]: {'dessert': 'pudding', 'entree': 'chicken', 'wine': 'chardonnay'}
当定义函数无法确定参数的具体个数时,可以使用可变参数来收集参数,动态参数有两种形式:一种是收集位置参数,另一种是收集关键字参数。
定义函数时使用*: 收集位置参数 (元组) 定义函数时使用**: 收集关键字参数 (字典) |
In [12]: def print_args(*args): #使用*收集位置参数
...: print('positional argument tuple:',args)
...:
In [13]: print_args()
positional argument tuple: ()
In [14]: print_args(1,2,3,4,5,6,'12131','abcd')
positional argument tuple: (1, 2, 3, 4, 5, 6, '12131', 'abcd')
In [15]: def print_kwargs(**args): #使用**收集关键字参数
...: print('Keyword arguments:',args)
...:
In [16]: print_kwargs(wine='merlot',entree='mutton')
Keyword arguments: {'wine': 'merlot', 'entree': 'mutton'}
-
函数作用域
Python创建、改变或查找变量名都是在名称空间中进行;在代码中变量名被赋值的位置决定了其能被访问的范围;函数定义了本地作用域,而模块定义了全局作用域。
每个模块都是一个全局作用域,因此,全局作用域的范围仅限于单个程序文件。每次对程序的调用都会创建一个新的本地作用域,赋值的变量除非声明为全局变量,否则均为本地变量。所有的变量名都可以归纳为本地、全局或者内置(由__builtin__模块提供)。
变量名引用分三个作用进行:首先本地变量、接着全局变量、最后内置变量。
In [17]: animal='fruitbat' #定义全局变量
In [18]: def print_global():
...: print('inside print_global:',animal) #调用全局变量
...:
In [20]: print_global()
inside print_global: fruitbat
In [21]: def print_global(): #当函数想改变全局变量时,执行函数会报错
...: print('inside print_global:',animal)
...: animal='wombat'
...: print('after the change:',animal)
...:
In [22]: print_global()
---------------------------------------------------------------------------
UnboundLocalError Traceback (most recent call last)
<ipython-input-22-e3810348dd19> in <module>()
----> 1 print_global()
<ipython-input-21-906f76821837> in print_global()
1 def print_global():
----> 2 print('inside print_global:',animal)
3 animal='wombat'
4 print('after the change:',animal)
5
UnboundLocalError: local variable 'animal' referenced before assignment
In [27]: def print_global(): #在函数内部声明全局变量,再更改时函数可以执行
...: global animal
...: print('inside print_global:',animal)
...: animal='wombat'
...: print('after the change:',animal)
...:
...:
...:
In [28]: print_global()
inside print_global: wolf
after the change: wombat
-
lambda()函数
在python中,lambda函数是用一个语句表达式的匿名函数,可以用它代替小的函数。
语法:lambda args: expression args: 以逗号分隔的参数列表 expression:用到 args 中各参数的表达式 |
Lambda 语句定义的代码必须是合法的表达式,不能出现多条件语句(可使用if的三元表达式)和其他非
表达式语句,如for和while等;Lambda 的首要用途是指定短小的回调函数;Lambda将返回一个函数而不
是将函数赋值给某个变量。
需要注意的是:Lambda 是一个表达式而不是语句;lambda是单个表达式,而不是代码块Def语句创建
的函数将赋值给某变量名,而lambda 表达式则直接返回函数Lambda也支持使用默认函数。
In [29]: f=lambda x,y,z:x+y+z #使用lambda匿名函数求和
...:
...:
In [30]: f(4,5,6)
Out[30]: 15
In [31]: f=lambda x,y,z=10:x+y+z
...:
...:
...:
In [32]: f(3,7)
Out[32]: 20
In [33]: stairs=['thus','meow','thud','hiss']
In [34]: f=lambda word: word.capitalize() +'!' #创建lambda函数将字符串大写
In [37]: for i in stairs:
...: print(f(i))
...:
...:
Thus!
Meow!
Thud!
Hiss!
-
函数式编程
也称作泛函编程, 是一种编程范型;他将电脑运算视作为数学上的函数计算,并且避免状态以及可变
数据;函数式编程语言最重要的基础是lamdba 演算,而且 lambda 演算的函数可以接受函数当作输入输
出。python支持有限的函数式编程:
filter(function or None, iterable) | 调用一个布尔函数function来迭代遍历每个iterable中的元素,返回一个使function返回值为真的元素序列。 |
map(func, *iterables) | 将函数func作用域给定序列的每一个元素,并用一个列表来提供返回值,如果function为none,func表现为一个身份函数,返回一个含有每个序列中元素集合的n个元组的列表。 |
reduce(func,seq[,init]) (python3中废弃) | 将二元函数作用域seq序列中的元素,每次携带一对(先前的结果以及下一个序列元素),连续地将现有的结果和下一个值作用在获得的随后结果上,最后减少序列为一个单一的返回值;如果出事值给定,第一个比较会是init和第一个序列元素而不是序列的两个头元素。 |
In [42]: li=[x for x in range(10) ]
In [43]: li
Out[43]: [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
In [44]: new_list=map(lambda a:a+10,li) #使用map函数对li列表中没个元素加10
In [45]: new_list
Out[45]: <map at 0x7fd595189828>
In [46]: list(new_list)
Out[46]: [10, 11, 12, 13, 14, 15, 16, 17, 18, 19]
In [48]: new_list2=filter(lambda arg:arg>5,li) #选出列表li中所有大于5的元素
In [49]: list(new_list2)
Out[49]: [6, 7, 8, 9]
-
函数的设计规范
耦合性:
(1)通过参数接受输入, 以及通过 return 产生输出以保证函数的独立性;
(2)尽量减少使用全局变量进行函数间通信;
(3)不要在函数中修改可变类型的参数;
(4) 避免直接改变定义在另外一个模块中的变量;
聚合性:
每个函数都应该有一个单一的、 统一的目标; 每个函数的功能都应该相对简单。
本文转自 梦想成大牛 51CTO博客,原文链接:http://blog.51cto.com/yinsuifeng/1911497,如需转载请自行联系原作者