轻松理解python中的闭包和装饰器 (下)
在 上篇 我们讲了python将函数做为返回值和闭包的概念,下面我们继续讲解函数做参数和装饰器,这个功能相当方便实用,可以极大地简化代码,就让我们go on吧!
能接受函数做参数的函数我们称之为高阶函数,例如filter, map, reduce这些函数
可以定义一个函数作为高阶函数例如:
def func(x, y, f):
return f(x)+f(y)
可以这样调用func(2,-1,abs) 函数返回结果为3
有些时候,我们不需要显式地定义传入的函数,直接传入匿名函数更方便。
在Python中,对匿名函数提供了支持。以map()函数为例,计算 f(x)=x2 时,除了定义一个f(x)的函数外,还可以直接传入匿名函数:
1 >>> map(lambda x: x * x, [1, 2, 3, 4, 5, 6, 7, 8, 9])
2 [1, 4, 9, 16, 25, 36, 49, 64, 81]
通过对比可以看出,匿名函数 lambda x: x * x 实际上就是:
def f(x):
return x * x
关键字lambda 表示匿名函数,冒号前面的 x 表示函数参数。
匿名函数有个限制,就是只能有一个表达式,不写return,返回值就是该表达式的结果。
使用匿名函数,可以不必定义函数名,直接创建一个函数对象,很多时候可以简化代码:
1 >>> sorted([1, 3, 9, 5, 0], lambda x,y: -cmp(x,y))
2 [9, 5, 3, 1, 0]
返回函数的时候,也可以返回匿名函数:
1 >>> myabs = lambda x: -x if x < 0 else x
2 >>> myabs(-1)
3 1
4 >>> myabs(1)
5 1
有些时候我们定义了一个函数,想在运行时动态增加功能,又不想改动函数本身的代码,这就可以用高阶函数实现: 可以接收一个函数作为参数对其包装,返回函数:
1 def f1(x):
2 return x*2
3
4 def new_fn(f):
5 def fn(x): # 定义一个内层函数
6 print "call: " + f.__name__ + "()"
7 return f(x) # 再内层函数中调用传递进来的函数
8
9 return fn # 返回内层函数
10
11 g1 = new_fn(f1) #进行调用
12 print g1(2)
执行上面代码,输出
call: f1()
4
可见我们将一个 f1 包装了一下,在它原有计算x*2的功能上又输出了自己的函数名。整个过程中我们定义的内层函数 fn 是关键,它在内部使用的是new_fn传递进来的参数 f ,这不就是前面讲的闭包吗? fn在添加其他功能后又调用了原来 f。
我们甚至可以这样
1 f1 = new_fn(f1)
2 print f1(2)
上面的代码输出结果一样,但是将f1传递进去又得到一个f1, 一样的调用方法但新的功能比以前的那个高级。
上面代码中f1 = new_fn(f1)这样的写法未免太过蹩脚,我们可以利用python提供的 装饰器 语法优雅的实现。上面new_fn的定义不变,我们只需要重新定义 f1 就可以了,如下:
1 @new_fn
2 def f1(x):
3 return x*2
4
5 @new_fn
6 def f2(x):
7 return x/2
8
9 print f1(2)
10 print f2(2)
输出如下:
call: f1()
4
call: f2()
1
我们在函数前面加了@new_fn就代表了对f1这个函数进行装饰加工,以后调用f1就直接调用了加工后的f1, 对于f2也这样,用起来一切很方便。
注意上面的代码只能对 f1 这样单个参数的函数能进行装饰,如果需要装饰两个参数的函数那new_fn就无法实现了,其实也很简单,我们只需要使用*args, **kw,就能保证任意个数的参数总是能正常调用,如下。
1 def new_fn(f):
2 def fn(*args, **kw): # 可以传入任意参数
3 print 'call ' + f.__name__ + '()...'
4 return f(*args, **kw)
5 return fn
6
7 @new_fn
8 def add(x, y):
9 return x + y
10
11
12 print add(1, 2)
其实仔细观察我们发现虽然调用add传递了参数1,2, 但是它只是传递到了装饰器中,也就是上面的 fn 中,我们可以随时在装饰器中改变它的参数然后再传递给 f。
经过上面的步骤现在总算是可以正常实现功能了。但是如果我们再提出需求呐(程序员总是害怕这一点
轻松理解python中的闭包和装饰器 (下)的更多相关文章
- 轻松理解python中的闭包和装饰器(上)
继面向对象编程之后函数式编程逐渐火起来了,在python中也同样支持函数式编程,我们平时使用的map, reduce, filter等都是函数式编程的例子.在函数式编程中,函数也作为一个变量存在,对应 ...
- python中的闭包和装饰器
重新学习完了函数,是时候将其中的一些重点重新捋一捋了,本次总结的东西只有闭包和装饰器 1.闭包 闭包是python函数中的一个比较重要功能,一般闭包都是用在装饰器上,一般学完闭包就会去学习装饰器,这俩 ...
- 21.python中的闭包和装饰器
python中的闭包从表现形式上定义(解释)为:如果在一个内部函数里,对在外部作用域(但不是在全局作用域)的变量进行引用,那么内部函数就被认为是闭包(closure). 以下说明主要针对 python ...
- Python 中的闭包与装饰器
闭包(closure)是函数式编程的重要的语法结构.闭包也是一种组织代码的结构,它同样提高了代码的可重复使用性. 如果在一个内嵌函数里,对在外部函数内(但不是在全局作用域)的变量进行引用,那么内嵌函数 ...
- python中的闭包与装饰器
#原创,转载请留言联系 装饰器的本质就是闭包,所以想知道装饰器是什么,首先要理解一下什么是闭包. 闭包 1. 外部函数返回内部函数的引用.2. 内部函数使用外部函数的变量或者参数. def outer ...
- 聊聊Python中的闭包和装饰器
1. 闭包 首先我们明确一下函数的引用,如下所示: def test1(): print("--- in test1 func----") # 调用函数 test1() # 引用函 ...
- python中函数总结之装饰器闭包
1.前言 函数也是一个对象,从而可以增加属性,使用句点来表示属性. 如果内部函数的定义包含了在外部函数中定义的对象的引用(外部对象可以是在外部函数之外),那么内部函数被称之为闭包. 2.装饰器 装饰器 ...
- 理解Python中的闭包
1.定义 闭包是函数式编程的一个重要的语法结构,函数式编程是一种编程范式 (而面向过程编程和面向对象编程也都是编程范式).在面向过程编程中,我们见到过函数(function):在面向对象编程中,我们见 ...
- 第十七篇 Python函数之闭包与装饰器
一. 装饰器 装饰器:可以拆解来看,器本质就是函数,装饰就是修饰的意思,所以装饰器的功能就是为其他函数添加附加功能. 装饰器的两个原则: 1. 不修改被修饰函数的源代码 2. 不修改被修饰函数的调用方 ...
随机推荐
- kmp算法理解与记录
字符串匹配的暴力解法 给定字符串s和p,寻找字符串p在字符串s中出现的位置,暴力解法如下所示: 如果当前字符匹配成功,++i;++j,继续匹配下一字符. 如果s[i]与s[j]匹配失败,令i-=(j- ...
- iOS + UIWebView 实践
1. 调用java script 现在只能实现弹出窗口 [self.m_webView stringByEvaluatingJavaScriptFromString:@"alert(1)&q ...
- Objective-C中的Hello World
Objective-C在C语言的基础之上,加入了自己的一些独特的特性,而且在Objective-C中兼容C语言原有的用法.在Objective-C中,实现Hello World和C语言有一定的区别,如 ...
- VPS 上ubuntu 里搭建VPN服务器
根据https://my.oschina.net/isnail/blog/363151里逐步完成,自己本机WIN10连接不成功,VM里面用WIN7连接也不行,找别人试连却成功了,然后自己用手机4G网络 ...
- Bmob 之 列表查询
查询列表,返回前20个数据: //创建BmobQuery实例,指定对应要操作的数据表名称 BmobQuery *query = [BmobQuery queryWithClassName:classN ...
- js框架Modernizr是什么东西? 他是前端开发HTML5和CSS3的强有力前端js检测类库
最近在研究modernizr的前端框架,发现这个Modernir对前端写页面非常友好,并且能够很快的建立起适应任何设备的html页面哦.在这里分享下基础教程,让大伙对modernizr是什么?做什么用 ...
- 阿里云服务器windows系统C盘一键清理脚本
@ECHO OFF @echo @echo @echo 清理几个比较多垃圾文件的地方 DEL /F /S /Q "C:\WINDOWS\PCHealth\ERRORREP\QSIGNOFF\ ...
- Unity 容器教程
文章摘自: http://www.cnblogs.com/qqlin/archive/2012/10/18/2720830.html
- UOJ #278. 【UTR #2】题目排列顺序(排序水题)
#278. [UTR #2]题目排列顺序 丢个传送门:http://uoj.ac/problem/278 描述 “又要出题了.” 宇宙出题中心主任 —— 吉米多出题斯基,坐在办公桌前策划即将到来的 U ...
- EF6的多线程与分库架构设计实现
1.项目背景 这里简单介绍一下项目需求背景,之前公司的项目基于EF++Repository+UnitOfWork的框架设计的,其中涉及到的技术有RabbitMq消息队列,Autofac依赖注入等常用的 ...