函数的高级应用

二、高阶函数

高级函数, 英文叫 Higher-order Function.

那么什么是高阶函数呢?

在说明什么是=高阶函数之前, 我们需要对函数再做进一步的理解!

2.1 函数的本质

函数的本质是什么?

函数和函数名到底是一种什么关系?


在python中,一切皆对象,那么函数也不例外,也是一种对象。

从本质上看,一个函数与一个整数没有本质区别,仅仅是他们的数据类型不同而已!

看下面的代码:

  1. def foo():
  2. pass
  3. print(foo) # 这里只打印了函数名, 并没有调用 foo 函数
  4. print(abs) # 直接打印内置函数, 而没有调用

说明:

  1. 从结果可以看出来, 直接把函数本身打印出来了, 自定义的函数与at后面的可以理解成函数在内存中的地址
  2. 如果是 python 内置函数会告诉你这是个内置函数.
  3. 你可以把函数想象中以前的数字一样, 仅仅表示内存中的一个对象.

函数名和函数的关系

其实函数名和以前的变量名没有本质的区别, 变量名是指向对象的一个符号, 那么函数名也是指向对象的一个符号.

动态把函数名赋值给新变量

函数名其实就是一个指向函数对象的变量.

那么我们是不是可以再创建一个变量也指向那个函数对象呢?

答案是肯定的!

  1. def foo():
  2. print("我是 foo 函数内的代码")
  3. a = foo
  4. print(a)
  5. print(foo)
  6. a()

说明:

  1. 你会发现直接打印a 和 foo的结果完全一样, 证明他们确实是指向了同一个函数对象
  2. 调用 a() 执行的就是foo函数内的代码. 因为他们其实就是一个对象.

给函数名重新赋值

既然函数名是变量名, 那么函数名也应该可以重新赋值!

  1. def foo():
  2. print("我是 foo 函数内的代码")
  3. foo = 3
  4. foo()

说明:

因为函数名已经被赋值为了整数,所以再调用就会抛出异常.

2.2 高阶函数

通过前面的了解, 我们已经知道函数名其实仅仅是一个普普通通的变量名而已.

那么是不是也意味着:函数也可以作为参数传递呢?

答案是肯定的!

一个可以接收函数作为参数的函数就是高阶函数!


一个最简单的高阶函数

  1. def foo(x, y, f): # f 是一个函数
  2. """
  3. 把 x, y 分别作为参数传递给 f, 最后返回他们的和
  4. :param x:
  5. :param y:
  6. :param f:
  7. :return:
  8. """
  9. return f(x) + f(y)
  10. def foo1(x):
  11. """
  12. 返回参数的 x 的 3次方
  13. :param x:
  14. :return:
  15. """
  16. return x ** 3
  17. r = foo(4, 2, foo1)
  18. print(r) # 72

说明:

  1. 这里的 foo 就是高阶函数, 因为他接收了一个函数作为参数.

  2. foo1作为参数传递给了foo, 而且foo中的局部变量f接收了foo传递过来的数据, 那么最终是foof同时指向了同一个对象.

总结

编写高阶函数,就是让函数的参数能够接收其他的函数。

把函数作为参数传入,这样的函数称为高阶函数,函数式编程就是指这种高度抽象的编程范式。

2.3 高阶函数另一种形式:把函数作为返回值

高阶函数除了可以接受函数作为参数外,还可以把函数作为结果值返回。

  1. def foo():
  2. x = 10
  3. def temp():
  4. nonlocal x
  5. x += 10 #x=x+10
  6. return x
  7. return temp
  8. f = foo()
  9. print(f())
  10. print(f())

说明:

  1. 调用foo()得到的一个函数, 然后把函数赋值给变量f, 这个时候ffoo内部的temp其实指向了同一个函数对象.
  2. 返回的函数每调用一次都会把foo 的局部变量x增加 10 .所以两次调用分别得到 20 和 30.
  3. 返回访问了外部函数的局部变量或者全局变量的函数,这种函数就是闭包.

2.4 内置高阶函数

高阶函数在函数式编程语言中使用非常的广泛.

本节简单介绍几个常用的高阶函数.

列表的排序, map/reduce, filter

2.4.1 排序sort()

2.41 sort()默认排序

到目前为止, 大家应该对列表已经比较熟悉了: 列表是有序, 允许重复.

注意:这里的有序是指的元素的添加顺序和迭代顺序一致.

但是我如果想对列表中的元素按照一定的规则排序该怎么做?

每个list实例都有有一个方法list.sort()可以帮我们完成这个工作.

sort() 默认对列表中的每个元素使用<进行比较,小的在前,大的在后.

也就是默认是升序排列


  1. nums = [20, 10, 4, 5, 3, 9]
  2. nums.sort()
  3. print(nums)


2.4.2 更改排序规则

比如, 列表中存储的是字符串, 大小写都有, 默认是按照字母表顺序来排列.

但是我们如果想忽略大小写的进行排列. 那么默认排序就无法满足我们的需求了

这个时候就需要用到key这个参数

key必须是一个函数, 则排序的时候, python 会根据这个函数的返回值来进行排序.

  1. ss = ["aa", "Aa", "ab", "Ca", "da"]
  2. def sort_rule(ele):
  3. return ele.lower()
  4. ss.sort(key=sort_rule)
  5. print(ss)


2.4.3 更改为降序

默认, 添加规则之后都是使用的升序排列.

如果需要降序排列, 则需要另外一个关键字参数 reverse

意思是问, 是否反序, 只要给 True 就表示降序了, 默认是 None


  1. ss = ["aa", "Aa", "ab", "Ca", "da"]
  2. def sort_rule(ele):
  3. return ele.lower()
  4. ss.sort(key=sort_rule, reverse=True)
  5. print(ss)


2.4.2 map()和filter()

函数编程语言通常都会提供map, filter, reduce三个高阶函数.

在python3中, map和filter仍然是内置函数, 但是由于引入了列表推导和生成器表达式, 他们变得没有那么重要了.

列表推导和生成器表达式具有了map和filter两个函数的功能, 而且更易于阅读.


2.4.2.1 map
  1. a = map(lambda x: x ** 2, [10, 20, 30, 40])
  2. print(list(a))
  3. print(type(a))

说明:

  1. map函数是利用已有的函数和可迭代对象生成一个新的可迭代类型:map类型
  2. map的参数1是一个函数, 参数2是一个可迭代类型的数据. map会获取迭代类型的每个数据, 传递给参数1的函数, 然后函数的返回值组成新的迭代类型的每个元素
  3. 也可以有多个迭代器, 则参数1的函数的参数个数也会增加.
  4. 新生成的迭代器类型的元素的个数, 会和最短的那个迭代器的元素的个数保持一致.
  1. a = map(lambda x, y: x + y, [10, 20, 30, 40], [100, 200])
  2. print(list(a))

使用列表推倒实现上面的功能

使用列表比map优雅了很多, 而且也避免了参数1的函数

  1. list1 = [10, 20, 30, 40]
  2. list3 = [x ** 2 for x in list1]
  3. print(list3)

  1. list1 = [10, 20, 30, 40]
  2. list2 = [100, 200]
  3. # 注意:列表推倒中这里是使用的笛卡尔积
  4. list3 = [x + y for x in list1 for y in list2]
  5. print(list3)


2.4.2.2 filter

对已有的可迭代类型起过滤作用, 然后生成新的可迭代类型

用法和map类型, 参数1也是函数, 会把当返回值为True的元素添加到新的可迭代类型中.

  1. list1 = [0, 1, 3, 4, 9, 4, 7]
  2. # 把奇数元素取出来
  3. print(list(filter(lambda x: x % 2 == 1, list1)))
  4. # 列表推倒的版本
  5. list2 = [x for x in list1 if x % 2 == 1]
  6. print(list2)

2.4.2.3 reduce

python 3中, reduce不再是直接的内置函数, 而是移到了模块functiontools内.

reduce的作用, 就是把一个可迭代序列的每一元素应用到一个具有两个参数的函数中.

例如:

reduce(lambda x, y: x+y, [1, 2, 3, 4, 5])就是计算((((1+2)+3)+4)+5)

  1. from functools import reduce
  2. def f(x, y):
  3. print("x=", x, "y=", y)
  4. return x + y
  5. """
  6. 参数1: 具有两个参数的函数
  7. 参数1:前面计算过的值 参数2:从可迭代类型中取得的新的值
  8. 参数2: 可迭代类型的数据
  9. 参数3: x的初始值, 默认是0
  10. """
  11. r = reduce(f, [1, 2, 3, 4, 5], 0)
  12. print(r) # 15

示例代码:使用reduce计算阶乘

  1. from functools import reduce
  2. def factorial(n):
  3. """计算n的阶乘
  4. :param n:
  5. :return:
  6. """
  7. return reduce(lambda x, y: x * y, range(1, n + 1), 1)
  8. print(factorial(5))
  9. print(factorial(6))
  10. print(factorial(7))

三、闭包

在函数编程语言中, 闭包是一个比较重要且强大的特性.

python 也支持闭包.

什么是闭包?

如果一个函数使用了外部函数的局部变量, 那么这个函数就是一个闭包.

闭包的特点:

  1. 闭包函数可以访问他所在的外部函数的局部变量. 即使外部函数已经运行结束, 对闭包函数来说仍然可以访问到外部函数的局部变量
  2. 闭包访问外部函数的局部变量的值, 总是这个变量的最新的值!

3.1.定义一个闭包

  1. def outer():
  2. x = 20
  3. def inner():
  4. """
  5. inner 函数访问了外部函数 outer 的局部变量 x, 所以这个时候 inner
  6. 就是一个闭包函数.
  7. :return:
  8. """
  9. nonlocal x
  10. x += 10
  11. return x
  12. x = 30
  13. return inner
  14. # 调用 outer, 得到的是内部的闭包函数 inner 所以 f 和 inner 其实指向了同一个函数对象
  15. f = outer()
  16. '''
  17. 调用 f. f是一个闭包函数,所以他访问的总是外部变量的最新的值,
  18. 所以 f 执行的时候 x 的值已经是30. 最终返回的是40
  19. '''
  20. print(f())


3.2.闭包的应用

闭包很强大, 也有一些比较适合的场景!


惰性求值(lazy evaluation, 延迟求值)

  1. def foo(msg):
  2. def say_msg():
  3. print("hello" + str(msg))
  4. return say_msg
  5. say = foo("志玲")
  6. say()

说明:

上面的代码中foo函数仅仅是声明了一个嵌套函数, 和把这个嵌套函数返回.

真正的代码其实是定义在了内部的嵌套函数中.

这种写法就是一种惰性求值!


使用闭包保持状态

如果需要在一系列函数调用中保持某种状态, 使用闭包是一种非常高效的方法.

一个简单的计数器:

  1. def count_down(num):
  2. def next():
  3. nonlocal num
  4. temp = num
  5. num -= 1
  6. return temp
  7. return next
  8. # 使用前面的函数计数
  9. next = count_down(10)
  10. while True:
  11. value = next() # 每调用一次就会减少一次计数
  12. print(value)
  13. if value == 0:
  14. break


还有一些其他应用, 比如装饰器

四、装饰器

装饰器也是应用闭包的一种场景.

什么是装饰器?

如果一个函数已经定义完成, 需要在不修改这个函数源码的前提下给这个函数增加一些功能, 这个时候就可以使用装饰器.

装饰器本质上是一个函数, 其主要用途是包装另一个函数或类.

包装的主要目的是透明地修改和增强被包装对象的行为.

4.1定义装饰器

装饰器可以用在方法上, 也可以用在类上.

目前我们只研究方法装饰器

其实装饰器和 java 中的注解有点像, 但是比 java 的注解容易使用了很多.


如果我们要给函数hello使用装饰器的方式增强功能, 语法如下:

  1. @strong
  2. def hello():
  3. print("我是 hello 函数中的代码")

说明:

  1. 在需要添加的装饰函数上面一行使用@来添加装饰器
  2. @后面紧跟中装饰器名strong, 当然你可以定于任何的名字.
  3. strong是装饰器, 本质上是一个函数. 他接收函数hello作为参数, 并返回一函数来替换掉hello(当然也可以不替换).
  4. hello使用装饰器之后, 相当于hello函数使用下面的代码被替换掉了.hello = strong(hello)
  5. 在调用hello的时候, 其实是调用的strong()返回的那个函数.

  1. def strong(fun): # fun 将来就是被替换的 hello
  2. def new_hello():
  3. print("我是装饰器中的代码, 在 hello 之前执行的")
  4. fun()
  5. print("我是装饰器中的代码, 在 hello 之后...执行的")
  6. return new_hello
  7. @strong
  8. def hello():
  9. print("我是 hello 函数中的代码")
  10. # 这里调用的其实是装饰器返回的函数.
  11. hello()


装饰器是语法糖

严格来讲, 装饰器只是语法糖.

装饰器就是一函数, 其参数是被装饰的函数.

综上, 装饰器的一大特性就是把装饰的函数替换成其他函数, 第二大特性就是装饰函数在加载模块的时候就立即执行了.

Python第六章-函数06-高阶函数的更多相关文章

  1. python基础编程: 编码补充、文件操作、集合、函数参数、函数递归、二分查找、匿名函数与高阶函数

    目录: 编码的补充 文件操作 集合 函数的参数 函数的递归 匿名函数与高阶函数 二分查找示例 一.编码的补充: 在python程序中,首行一般为:#-*- coding:utf-8 -*-,就是告诉p ...

  2. Python之路-函数基础&局部变量与全局变量&匿名函数&递归函数&高阶函数

    一.函数的定义与调用 函数:组织好的.可重复使用的.用户实现单一或者关联功能的代码段.函数能够提高应用的模块性和代码的重复利用率.Python提供了很多内置的函数,比如len等等,另外也可以根据自己的 ...

  3. Python小世界:匿名函数、高阶函数、推导式

    前言 木子本人搞起Python已有多年,一直觉得什么都会,但是有时候实操起来,才觉得很多底层基础的知识都没有彻底的灵活掌握. 另外,网上关于Python基础知识的各种普及已有太多太多. 附上相关大神的 ...

  4. python学习8—函数之高阶函数与内置函数

    python学习8—函数之高阶函数与内置函数 1. 高阶函数 a. map()函数 对第二个输入的参数进行第一个输入的参数指定的操作.map()函数的返回值是一个迭代器,只可以迭代一次,迭代过后会被释 ...

  5. Python 函数进阶-高阶函数

    高阶函数 什么是高阶函数 高阶函数就是能够把函数当成参数传递的函数就是高阶函数,换句话说如果一个函数的参数是函数,那么这个函数就是一个高阶函数. 高阶函数可以是你使用def关键字自定义的函数,也有Py ...

  6. 12、python中的函数(高阶函数)

    一.高阶函数 函数实际上也是一个对象,所以也能由变量指向一个函数对象,实际上函数名就是一个变量名.那么函数是传入变量作为参数的,如果传入的变量指向的是函数对象,这种函数就叫高阶函数. 高阶函数就是传入 ...

  7. Python技法3:匿名函数、回调函数和高阶函数

    1.定义匿名或内联函数 如果我们想提供一个短小的回调函数供sort()这样的函数用,但不想用def这样的语句编写一个单行的函数,我们可以借助lambda表达式来编写"内联"式的函数 ...

  8. 初识python:高阶函数(附-高阶函数)

    定义: 变量可以指向函数,函数的参数能接收变量,那么,一个函数可以接收另一个函数作为参数,这种函数就称之为高阶函数. 简单说就是:把函数当作参数传递的函数就是高阶函数 特性 1.把一个函数名当作实参传 ...

  9. python3笔记十六:python匿名函数和高阶函数

    一:学习内容 lambda函数 map函数与reduce函数 filter函数 sorted函数 二:匿名函数-lambda 1.概念:不使用def这样的语句去定义函数,使用lambda来创建匿名函数 ...

  10. python六十四课——高阶函数练习题(二)

    总结:高阶函数以及匿名函数之间的配合使用 from functools import reduce #模块一:lambda和filter的结合使用 #lt = [1,2,3,4,5,6,7,8,9] ...

随机推荐

  1. Flutter Widgets 之 ListWheelScrollView

    注意:无特殊说明,Flutter版本及Dart版本如下: Flutter版本: 1.12.13+hotfix.5 Dart版本: 2.7.0 基础用法 在展示大量数据的时候我们第一会想到使用ListV ...

  2. d3.js ---画坐标轴

    画坐标轴 //使用d3的svg的axis()方法生成坐标轴 var x_axis = d3.svg.axis().scale(scale_x), y_axis = d3.svg.axis().scal ...

  3. 【JavaScript】DOM之表单操作

    DOM 表单操作 1.获取表单 获取表单元素 以Document对象中forms属性来获取当前HTML页面所有表单集合以Document对象中表单的name属性值来获取表单元元素 <body&g ...

  4. ionic监听android返回键(实现“再按一次退出”功能)

    在android平台上的app,在主页面时经常会遇到"再按一次退出app"的功能,避免只按一下返回键就退出app提升体验优化. 1.这个功能需要我们用到ionic提供的regist ...

  5. webpack从0到1超详细超基础学习教程

    概念 自己是一个一听到webpack就头大,看着一堆不知道那是什么玩意的东西总觉得自己做好前端就行了,但是在使用vue-cli的时候总觉得要改其中的一些东西进行项目初始化的时候能够更好使用!所以想要根 ...

  6. 必备技能五、router路由钩子

    在路由跳转的时候,我们需要一些权限判断或者其他操作.这个时候就需要使用路由的钩子函数. 定义:路由钩子主要是给使用者在路由发生变化时进行一些特殊的处理而定义的函数. 总体来讲vue里面提供了三大类钩子 ...

  7. ios background task

    今天要实现一个需求,当用户触摸HOME键,将应用切换到后台时,启动自动备份的任务.这涉及到ios的后台任务处理,本文简单总结一下 首先,ios app有5种状态,分别是:not running, in ...

  8. ubuntu1804自带的vim和vi都是用什么版本?

    之前搜索vim一些命令时,经常看到有人说ubuntu自带的vim是是vim.tiny的,功能不全. 什么需要先卸载,再重装,真的是这样吗? 我查了一下,vim的版本号 vim --version vi ...

  9. ubunto python + vnstat 限制每天流量使用 使用iptables

    上次想使用 iptables 转发80 端口,试了一段时间,没有成功.哪位知道是什么原因,还麻烦告诉我. 这次使用 iptables 禁用 80 443 出站,经过试验可以成功. 通过 iptable ...

  10. Simulink仿真入门到精通(五) Simulink模型的仿真

    5.1 模型的配置仿真 由各种模块所构建的可视化逻辑连接,只是模型的外在表现,模型仿真的核心驱动器是被称作解算器(Solver)的组件,相当于Simulink仿真过程的心脏,驱动着模型仿真,它在每一个 ...