其他关于Python的总结文章请访问:https://www.jianshu.com/nb/47435944
Python中使用函数作为参数传递以及将函数作为参数返回--函数式编程
所有定义的函数其实它的名字都是指向这个函数本身的一个变量,也就是说如果一个函数 def mySum(s,y)
,使用 mySum(1,2)
是调用这个函数,而 mySum
本身是一个变量,和其他任何变量一样,是可以赋值、传递等操作的。
函数变量的赋值
可以通过赋值操作把一个指向函数的变量(也就是函数名)赋给另一个变量让他也指向这个函数:
def mySum(x, y):
return (x + y)
anotherFuc = mySum
print(anotherFuc(1,2))
print(mySum(1,2))
print(mySum is anotherFuc)
print(mySum == anotherFuc)
------
3
3
True
True
当然也可以将其他的变量赋给一个函数名,但是这样通常没有意义,因为赋值后原来指向函数的变量(也就是函数名)有了新的意义,也就无法再作为函数名来调用了:
mySum = 10
print(mySum)
result = mySum(1,2)
------
10
Traceback (most recent call last):
result = mySum(1,2)
TypeError: 'int' object is not callable
函数作为参数传递
函数变量和其他变量一样可以作为参数传入另一个函数,因为传递的是参数,所以传递时不要跟上后边的括号以及可能存在的参数,只要传递函数名即是传递了函数变量:
def fr5():
return 5
def fr4():
return 4
def printfr(f1, f2):
print(f1())
print(f2())
printfr(fr4,fr5)
------
4
5
函数变量作为返回值
函数可以作为变量像其他变量一样使用 return
返回,比如我们定义一个函数 f
,在它里边定义一个函数 g
,然后可以返回 g
这个函数:
def f(*args):
def g():
sum = 0
for i in args:
sum+=i
return sum
return g
g1 = f(1,2,3,4,5)
g2 = f(6,7,8,9,10)
print(g1 == g2)
print(g1 is g2)
print(g1())
print(g2())
------
False
False
15
40
可以看到,在一个接收 args
自定参数的函数 f
内部,定义了一个没有参数的函数 g
,g
却使用了 f
的参数,然后返回了这些参数的和,而 f
函数则把 g
这个函数变量返回了。我么可以将返回的值赋给新的变量,也就是 g1
和 g2
,并且 g1
和 g2
是没有关系的,及不相等也不等同,再次调用 g1
和 g2
才是调用了 f
返回的函数 g
。
这里可以看到有一个很好的效果是:内部的函数 g
没有任何传入参数,但是由于它被定义在了函数 f
内部,所以它可以随意使用f的所有东西,包括传入 f
的参数、f
定义的局部变量等,而 f
由于返回了函数 g
,其实也就将自己的一些信息比如自己的参数和自己定义的变量(当然这里没有)等都又保留在了 g
中返回。这种强大的程序结构称为闭包(Closure)
闭包中的循环
闭包中遇到循环是一个比较麻烦的事情,直接看一个例子:
def f():
fr = []
for i in range(1,4):
def g():
print(i*i)
fr.append(g)
return fr
[f1,f2,f3] = f()
f1()
f2()
f3()
------
9
9
9
在外层 f
函数中,我们创立了一个循环,每一个循环中都创建了一个子函数 g
,而且将 g
推入了一个数组中,最后的最后,循环结束后,返回这个装有三个函数 g
的数组。在外边以此调用 f1
,f2
和 f3
会发现居然都打印了9,也就是 i=3
的情况,这就是闭包中使用循环的问题,其原因在于:
- 内层函数会用到外层函数的一些变量,而这些变量是跟外层函数有关的,外层函数的变量改变了,内部用到的也会变
- 函数所表现的结果是调用时产生的,而不是赋值时
也就是说,使用 f1()
调用这个f1函数的时候,它打印了9,而不是在 f1
被赋值的时候,来看一下调用f1
的时候,其实f1
是由 g
赋值而来的,也就是调用g
,再看 g
函数的内容:print(i * i)
,很显然,调用 f1
的时候,i
等于3,因为那个循环早就进行完了,i
已经等于3了,尽管在发生 fr.append(g)
这个赋值操作时 i
还是1,但是这没有什么作用,只有在调用函数时才是真正用到的时候,而此时调用的函数只做了一个操作:print(i * i)
,自然是打印此时 i
值的平方,跟赋值时 i
等于几无关。
所以如果要达到另一种效果只好这么做:
def f():
def l(j):
def g():
print(j * j)
return g
fr = []
for i in range(1,4):
fr.append(l(i))
return fr
[f1,f2,f3] = f()
f1()
f2()
f3()
------
1
4
9