From pixabay
函数是python编程中不可或缺的一部分。好的自定义函数可以使代码更加模块化、规范化。
一、函数基础
函数就是为了实现某一功能的代码块,只要函数写好了,就可以被重复利用,我们来看下面这个例子:
defmy_func(message):
print('Got a message: {}'.format(message))
return1
# 调用函数 my_func()
my_func('Hello World')
# 输出
Gota message:HelloWorld
其中:
def 是函数的声明
my_func是函数的名称
message是函数的参数
print这一行是函数的实体,可以是相应的执行语句
函数的最后可以返回调用结果(return,yield)
大概是下面这种形式:
defname(param1,param2,...,paramN):
statements
return/yieldvalue# optional
python函数可以设置默认值,如下:
deffunc(param=0):
...
这样在调用func()函数的时候,如果没有传递参数,则参数的默认值就为0,如果传入了参数,则param就会被覆盖了为传入的值。python与其他语言相比的一大特点是:python是dynamically typed,可以接受任何数据类型。对于函数参数来说一样适用,如下:
defmy_sum(a,b):
returna+b
result=my_sum(3,5)
print(result)
# 输出
8
上面的传递两个数值,其实传递两个列表也是可以的,表示两个列表相连接:
print(my_sum([1,2],[3,4]))
# 输出
[1,2,3,4]
同理,传递字符串也是可以的,表示字符串的拼接:
print(my_sum('hello ','world'))
# 输出
hello world
但是,如果两个参数的数据类型不同,比如一个是列表,一个是字符串,这两者是无法相加的,那就会报错,如下:
print(my_sum([1,2],'hello'))
TypeError:can only concatenate list(not"str")to list
由上我们可以看到,python不用考虑输入的数据类型,而是将其交给具体的代码和解释器去判断,同样的一个函数,可以应用在整型,列表,字符串等等的操作中。在python语言中,我们把这种行为称为多态。python的另一个特性是支持函数的嵌套,所谓的函数的嵌套,就是函数里面又定义了函数,如下:
deff1():
print('hello')
deff2():
print('world')
f2()
f1()
# 输出
hello
world
上面的例子中函数f1()中又嵌套了一个函数f2(),在调用f1()的时候,会先打印字符串"hello",然后再在f1()中调用f2(),打印字符串"world"。函数的嵌套的主要所用如下:(1)、函数的嵌套可以保证内部函数的隐私,只能被外部函数所调用和访问,不会暴露在全局作用域,因此,有一些隐私数据不想暴露在外,这时候可以使用函数的嵌套。如下:
defconnect_DB():
defget_DB_configuration():
...
returnhost,username,password
conn=connector.connect(get_DB_configuration())
returnconn
这里的getDBconfiguration就是内部函数,它不会被除connect_DB以外的程序调用,如下直接调用是会报错的:
get_DB_configuration()
# 输出
NameError:name'get_DB_configuration'isnotdefined
(2)、合理的使用嵌套,可以提高代码的运行效率。如下:
deffactorial(input):
# validation check
ifnotisinstance(input,int):
raiseException('input must be an integer.')
ifinput<0:
raiseException('input must be greater or equal to 0')
...
definner_factorial(input):
ifinput<=1:
return1
returninput*inner_factorial(input-1)
returninner_factorial(input)
print(factorial(5))
上面的例子是递归计算一个数的阶乘,在计算之前需要检查输入是否合法,如果不使用函数嵌套,每一次递归的时候都要检查一次,大大降低了代码的运行效率。如果使用了递归,那么只会检查一次。
二、函数变量的作用域
python中变量的作用域和其他语言类似。如果变量在函数内部定义,就称为局部变量,只有在函数内部有效,一旦函数执行完毕,局部变量会被回收,无法访问,如下:
defread_text_from_file(file_path):
withopen(file_path)asfile:
...
我们内部定义的fie这个变量,只有在函数内部有效,在外部是无法访问的。相应的,全局变量是定义在整个文件层次上面的,如下:
MIN_VALUE=1
MAX_VALUE=10
defvalidation_check(value):
ifvalueMAX_VALUE:
raiseException('validation check fails')
MINVALUE和MAXVALUE是全局变量,在任意地方都可以调用。但是我们不能在函数内部随意更改全局变量的值。比如下面的写法就会有问题:
MIN_VALUE=1
MAX_VALUE=10
defvalidation_check(value):
MIN_VALUE+=1
validation_check(5)
Traceback(most recent call last):
File"",line1,in
File"",line2,invalidation_check
UnboundLocalError:local variable'MIN_VALUE'referenced before assignment
这是因为python解释器会默认内部变量为局部变量,但是在局部又发现MIN_VALUE并没有声明,因此就无法操作,如果想要在函数内部操作全局变量的值,则需要global声明,如下:
MIN_VALUE=1
MAX_VALUE=10
defvalidation_check(value):
globalMIN_VALUE
MIN_VALUE+=1
validation_check(5)
这里的global关键字并不是表示重新创建了一个变量,而是告诉python解释器,MIN_VALUE就是全局变量,可以对其进行操作。但是,如果局部变量和全局变量同名,那么在函数内部,局部变量会覆盖全局变量,如下:
MIN_VALUE=1
MAX_VALUE=10
defvalidation_check(value):
MIN_VALUE=3
类似的,对于嵌套函数来说,内部函数可以访问外部函数的变量,但是无法对其进行修改,如果要修改,则要加上nonlocal关键字,如下:
defouter():
x="local"
definner():
nonlocalx# nonlocal 关键字表示这里的 x 就是外部函数 outer 定义的变量 x
x='nonlocal'
print("inner:",x)
inner()
print("outer:",x)
outer()
# 输出
inner:nonlocal
outer:nonlocal
如果不加nonlocal关键字,内部函数变量又和外部函数变量同名,那么同样的,内部函数变量会覆盖外部函数的变量,如下:
defouter():
x="local"
definner():
x='nonlocal'# 这里的 x 是 inner 这个函数的局部变量
print("inner:",x)
inner()
print("outer:",x)
outer()
# 输出
inner:nonlocal
outer:local
三、闭包
闭包其实和嵌套函数类似,不同的是嵌套函数返回的是一个值,而闭包返回的是一个函数,返回的函数通常赋予一个变量,这个变量可以被执行调用。比如我们想计算一个数的n次幂,用闭包写成如下:
defnth_power(exponent):
defexponent_of(base):
returnbase**exponent
returnexponent_of# 返回值是 exponent_of 函数
square=nth_power(2)# 计算一个数的平方
cube=nth_power(3)# 计算一个数的立方
square
# 输出
.exponent(base)>
cube
# 输出
.exponent(base)>
print(square(2))# 计算 2 的平方
print(cube(2))# 计算 2 的立方
# 输出
4# 2^2
8# 2^3
这里的外部函数nthpower()的返回值是exponentof()的函数名,而不是一个具体值,需要注意的是在执行square = nthpower(2)和cube = nthpower(3),外部函数nthpower的参数exponent仍然会被内部exponentof()函数记住,这样我们调用square(2)和cube(2),程序会顺利的输出结果。闭包可以让程序变得更简洁易读,并且可以减少调用导致的不必要的开销,增强程序运行效率。
四、总结
(1)、python中函数的参数可以接受任意数据类型,使用起来需要注意,必须是在函数开头加入数据类型检查;(2)、python中的函数参数可以设置默认值;(3)、嵌套函数的使用,可以保证一定数据的隐私性,提高程序的运行效率;(4)、合理的使用闭包,可以降低程序的复杂性,提高可读性;