Python中使用函数作为参数传递和返回--函数式编程

其他关于Python的总结文章请访问:https://blog.csdn.net/qq_38962621/category_10299380.html

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 内部,定义了一个没有参数的函数 gg 却使用了 f 的参数,然后返回了这些参数的和,而 f 函数则把 g 这个函数变量返回了。我么可以将返回的值赋给新的变量,也就是 g1g2 ,并且 g1g2 是没有关系的,及不相等也不等同,再次调用 g1g2 才是调用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 的数组。在外边以此调用 f1f2f3 会发现居然都打印了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
  • 12
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
函数式编程是一种编程范式,它将计算视为数学函数的求值过程。Python作为一门多范式的编程语言,也支持函数式编程。在函数式编程函数被视为一等公民,可以作为参数传递给其他函数,也可以作为返回返回函数式编程强调不可变性和无副作用,通过使用函数来实现。 在Python函数式编程可以通过以下几个特性来实现: 1. 高阶函数Python函数可以作为参数传递给其他函数,也可以作为返回返回。这使得我们可以使用高阶函数来实现函数的组合、柯里化等功能。 2. 匿函数Python的lambda表达式允许我们创建匿函数,这些函数可以在需要时直接定义,而不需要使用def关键字进行命。 3. map、filter和reduce函数:这些内置函数函数式编程的重要工具。map函数可以将一个函数应用于一个可迭代对象的每个元素,并返回一个新的可迭代对象;filter函数根据指定的条件过滤可迭代对象的元素,并返回一个新的可迭代对象;reduce函数对可迭代对象的元素进行累积操作,返回一个单一的结果。 4. 列表推导式和生成器表达式:这些语法结构允许我们使用简洁的方式创建新的列表或生成器,从而实现对可迭代对象的转换或筛选。 5. 不可变性:函数式编程强调不可变性,即函数参数和局部变量函数执行期间不可修改。Python的元组和frozenset等不可变对象可以用来实现这一特性。 6. 递归:函数式编程常常使用递归来解决问题,通过函数自身调用来实现循环和迭代的效果。 总结起来,Python函数式编程提供了一系列工具和语法结构,使得我们可以更加方便地使用函数作为一等公民,并且实现函数的组合、柯里化、惰性求值等特性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值