python中sort返回值_Python函数你真的都学会了吗?来看看这篇Python高阶函数!

26c733ac2153c1fcab2a08e53247e4e8.png

二、高阶函数

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

那么什么是高阶函数呢?

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

2.1 函数的本质

函数的本质是什么?

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

87abcca5e6ccb51978ea5eeb4b4887b8.png

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

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

看下面的代码:

def foo():    passprint(foo)  # 这里只打印了函数名, 并没有调用 foo 函数print(abs)  # 直接打印内置函数, 而没有调用
6a676fd29cb6a4d6e22a14d494018885.png

说明:

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

函数名和函数的关系

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

585552dc735c10cf8f2b05abe3a54fcf.png

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

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

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

答案是肯定的!

857e1ae6a370a3f2be9b0a0687dac147.png
def foo():    print("我是 foo 函数内的代码")a = fooprint(a)print(foo)a()
8a81ef28204b45e471e10d5e8e2fd90c.png
784af5c77e916c791830e046fb723ff8.png

说明:

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

给函数名重新赋值

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

def foo():    print("我是 foo 函数内的代码")foo = 3foo()
ab92d35dd5b448ff1316f39b0a75ece4.png

说明:

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

2.2 高阶函数

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

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

答案是肯定的!

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


一个最简单的高阶函数

def foo(x, y, f):  # f 是一个函数    """    把 x, y 分别作为参数传递给 f, 最后返回他们的和    :param x:    :param y:    :param f:    :return:    """    return f(x) + f(y)def foo1(x):    """    返回参数的 x 的 3次方    :param x:    :return:    """    return x ** 3r = foo(4, 2, foo1)    print(r)    # 72

说明:

  1. 这里的 foo 就是高阶函数, 因为他接收了一个函数作为参数.
  2. foo1作为参数传递给了foo, 而且foo中的局部变量f接收了foo传递过来的数据, 那么最终是foo和f同时指向了同一个对象.
978d650ffbf4d0ff61e98c13e852111a.png

总结

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

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

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

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

def foo():    x = 10    def temp():        nonlocal x        x += 10 #x=x+10        return x    return tempf = foo()print(f())print(f())
bb3384229ba9cee9f9a89a82723c6f2d.png

说明:

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

2.4 内置高阶函数

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

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

列表的排序, map/reduce, filter等

2.4.1 排序sort()

2.41 sort()默认排序

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

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

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

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

sort() 默认对列表中的每个元素使用

也就是默认是升序排列


nums = [20, 10, 4, 5, 3, 9]nums.sort()print(nums)
753fccf0676ce98ec0274b3a75c40d9b.png

2.4.2 更改排序规则

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

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

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

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

ss = ["aa", "Aa", "ab", "Ca", "da"]def sort_rule(ele):    return ele.lower()ss.sort(key=sort_rule)print(ss)
c29d74fdf65d8e05ffd58bae3da0a07a.png

2.4.3 更改为降序

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

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

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


ss = ["aa", "Aa", "ab", "Ca", "da"]def sort_rule(ele):    return ele.lower()ss.sort(key=sort_rule, reverse=True)print(ss)
6fc0769360ca3d48d4da3161f093458f.png

2.4.2 map()和filter()

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

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

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


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

说明:

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

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

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

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

2.4.2.2 filter

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

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

list1 = [0, 1, 3, 4, 9, 4, 7]# 把奇数元素取出来print(list(filter(lambda x: x % 2 == 1, list1)))# 列表推倒的版本list2 = [x for x in list1 if x % 2 == 1]print(list2)
fc5223f72c62b5548a0401eabd093beb.png
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)

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

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

from functools import reducedef factorial(n):    """计算n的阶乘    :param n:    :return:    """    return reduce(lambda x, y: x * y, range(1, n + 1), 1)print(factorial(5))print(factorial(6))print(factorial(7))
cdf8374dbb909f5daf45210fed049928.png

三、闭包

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

python 也支持闭包.

什么是闭包?

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

闭包的特点:

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

3.1.定义一个闭包

def outer():    x = 20    def inner():        """        inner 函数访问了外部函数 outer 的局部变量 x, 所以这个时候 inner        就是一个闭包函数.        :return:        """        nonlocal x        x += 10        return x    x = 30    return inner# 调用 outer, 得到的是内部的闭包函数 inner   所以 f 和 inner 其实指向了同一个函数对象f = outer()''' 调用 f. f是一个闭包函数,所以他访问的总是外部变量的最新的值, 所以 f 执行的时候 x 的值已经是30. 最终返回的是40'''print(f())
99743b151a4dc112967119d85f8a4699.png

3.2.闭包的应用

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


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

def foo(msg):    def say_msg():        print("hello" + str(msg))    return say_msgsay = foo("志玲")say()

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

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

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


使用闭包保持状态

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

一个简单的计数器:

def count_down(num):    def next():        nonlocal num        temp = num        num -= 1        return temp    return next# 使用前面的函数计数next = count_down(10)while True:    value = next()  # 每调用一次就会减少一次计数    print(value)    if value == 0:        break
cc4dbe8dbc026c748e426c7f9726836b.png

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

四、装饰器

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

什么是装饰器?

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

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

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

4.1定义装饰器

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

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

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


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

@strongdef hello():    print("我是 hello 函数中的代码")

说明:

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

def strong(fun):  # fun 将来就是被替换的 hello    def new_hello():        print("我是装饰器中的代码, 在 hello 之前执行的")        fun()        print("我是装饰器中的代码, 在 hello 之后...执行的")    return new_hello@strongdef hello():    print("我是 hello 函数中的代码")# 这里调用的其实是装饰器返回的函数.hello()
db6200f3a1d6985e42e62c546501bb9b.png

装饰器是语法糖

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

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

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/267503.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

基于数据库的事务消息解决分布式事务方案

转载请注明出处:http://www.cnblogs.com/lizo/p/8516502.html 概述 当单库已不能支撑当前业务的时候,我们往往都考虑进行分库(横向拆分或者纵向拆分)。但分库有个无法回避的问题,就是事务问题。网上有很多分布式事务解…

中service层的作用_浅析Java中dto、dao、service、controller的四层结构

目前我所在的项目组采用的是SpringBoot框架,前端使用BootStrapjQuery。SpringBoot是BS开发框架之一,不用单独开启tomcat服务器,目前比较流行,一般开发大型项目时会将所有的功能细分为许多小模块,每个模块都有dto、dao、…

【iOS开发必收藏】详解iOS应用程序内使用IAP/StoreKit付费、沙盒(SandBox)测试、创建测试账号流程!...

Himi 原创, 欢迎转载,转载请在明显处注明! 谢谢。 原文地址:http://blog.csdn.net/xiaominghimi/article/details/6937097 终于在11月公司的游戏即将上线了,那么对于iOS游戏来说当今都是内置道具收费属于主流&#xf…

不越狱换壁纸_终于来了!iOS 14.3 正式版,可自动定时换壁纸

嘿嘿,我没有猜错吧!iOS 14.3 正式版会在12月15日凌晨时段发布,在前几天我就有提到,这一天会发布,主要是与新品 AirPods Max 发售时间与iOS 14.3正式版发布时间一致。其次这次发布iOS 14.3正式版更新内容与 iOS 14.3 RC…

Hibernate【inverse和cascade属性】知识要点

Inverse属性 Inverse属性:表示控制权是否转移.. true:控制权已转移【当前一方没有控制权】false:控制权没有转移【当前一方有控制权】Inverse属性,是在维护关联关系的时候起作用的。只能在“一”的一方中使用该属性!Inverse属性的…

中list如何清空_如何根据索引删除 list 中的元素

这个问题很简单, 首先想到的就是a list(range(10)) del a[2]这个就可以很方便的删除掉 a 中的第 3 个元素.如果我想删除多个元素怎么办, 比如我想删除第 3, 4, 5, 6 个元素? 这个也很好办:a list(range(10)) del a[2:6]那么我要删除的元素的索引不连续呢? 比如我要删除第 3…

查询时拼接两列数据_如何用VBA代码查询两列数据差异?

爱就一个字,我只说一次……北京市第三交通委提醒您:代码千万条,注释第一条,命名不规范,修订两行泪……咳,给大家拜晚年了,再提前祝大家元宵快乐……我们今天和大家分享的内容是如何用VBA代码查询…

setTimeout详解

https://www.cnblogs.com/wzndkj/p/7069331.html 一、setTimeout基础 setTimeout(func|code,delay);第一个参数表示将要推迟的函数名或者一段代码,第二个参数表示推迟执行的毫秒数eg: console.log(1); setTimeout(console.log(2),1000); console.log(3);answer: 1 3…

Kappa电商负责人顾皓澜:电商业务一直保持盈利

Kappa电商负责人顾皓澜(TechWeb配图) 【TechWeb报道】10月初,闹得沸沸扬扬的淘宝商城事件吸引了无数互联网用户的目光,处于漩涡中心的淘宝商城当仁不让地挤入话题排行榜,商城上大小卖家的命运也牵动了众多消费者的心。…

Citrix VDI-in-a-Box 第二篇:架构篇

前言:为什么Citrix会收购Kaviza,就是因为其VDI-in-a-Box产品架构比较简单。 本文重点描述其架构和安装要求。 如果你想了解一个东西,首先必须了解其架构。 vdiManager是管理整个架构的工具,所有的虚拟机都运行在Hypervisor上。整 …

看完此文再不懂区块链算我输,用Python从零开始创建区块链

如果你还没有听说过 3 点钟区块链群,说明你还不是链圈的人;如果你还没有加入 3 点钟区块链群,说明你还不是链圈的大佬;如果你还没有被 3 点钟区块链群刷屏,说明你还体会不到什么是“币圈一天,人间一年”。 …

重新加一个window_Activity、View、Window关系,进程间通信,责任链模式,Https,数据存储...

码仔,今天就给大家带来了《每日一道面试题》的第九期:01理解Activity View window的关系 Activity像一个工匠(控制单元),Window像窗户(承载模型),View像窗花(显示视图)LayoutInflater像剪刀,Xml配置像窗花图纸。 Activ…

排他网关(ExclusiveGateWay)

网关(ExclusiveGateWay) 作者:邓家海2018年3月11日 00:13:25 情景:某一家公司最近在给一个单位做一个财务审批的OA。具体需求是这样的:当部门申请的金额小于一万块的时候,财务部可以直接决策。当部分申请的金额大于一万…

离线安装宝塔lnmp_宝塔LNMP环境 Nginx安装EduSoho教程说明

[toc]宝塔下使用LNMP Nginx安装EduSoho创建站点 宝塔后台 > 网站 > 添加站点 > 输入信息 > 提交填写信息创建完成设置运行目录 宝塔后台 > 网站 > 管理运行目录选择web目录后保存 网站目录 > 运行目录 > 保存修改配置文件 配置文件 > 修改参数 >…

NHibernate初学者指南(10):一级和二级缓存

一级缓存 为了获得更好的性能,NHibernate智能地缓存数据。NHibernate有不同的缓存机制起作用,最重要的就是一级缓存。每个session对象维持一个一级缓存,session对象创建时缓存创建,session对象释放时缓存销毁。 缓存只不过是一个哈…

Freemarker模板引擎

模板引擎的实质就是将页面结构提前写好,然后将数据渲染到模板上生成一个静态页面,这样一来,下次就可以 直接访问静态文件,不用进行额外的获取数据的操作(例如:访问数据库),这样大大提…

postgresql主从备份_基于windows平台的postgresql主从数据库流备份配置

基于windows平台的postgresql主从数据库流备份配置因工作需要,需要搞pg数据库的主从备份,领导给了个方向使用流备份,于是开始朝着这个方向进发。鸣谢大佬A_ccelerator的博客一、配置主从库1.环境准备对于 pg 的主从库配置,建议是使…

msvcrt.lib和LIBCD.lib链接冲突

今天在移植一个开源代码到windows的VC6工程,编译时出现了这些奇怪的LINK错误。 msvcrt.lib(MSVCRT.dll) : error LNK2005: _toupper already defined in LIBCD.lib(toupper.obj)msvcrt.lib(MSVCRT.dll) : error LNK2005: _tolower already defined in LIBCD.lib(to…

jq获取最后一个子节点_如何选择jQuery中的最后一个子元素?

牧羊人nacy如果要选择最后一个子元素,并且需要具体说明元素类型,则可以使用选择器last-of-type这是一个例子:$("div p:last-of-type").css("border", "3px solid red");$("div span:last-of-type").…

面向对象 - 继承/组合 - 总结

面向对象 - 继承:1.继承: 类与类之间的关系 什么是什么的关系 eg:人是动物 狗是动物 功能: 解决代码重用问题, 创建新类的方式, 类: 可继承一个或多个父类: 父类 基类/超类 类 派生类/子类 类: 对象之间相似的特征 父类:…