Python学习_08_函数式编程
在python中,函数名也是一个变量,代表对一个函数内容的引用,意味着可以作为参数传入到其他函数中,根据这个特性,发散出装饰器、闭包等概念,并涉及到变量作用域等问题。
函数
python中函数操作符为(),在任何需要调用函数的地方都需要在函数名后面加(),表示调用该函数,否则的话仅仅表示一个函数对象,当然可以保存这个对象,或者将该函数对象传入到其他函数中,以供延迟调用。
函数的参数分为按顺序确定的位置参数、默认参数、非关键字可变长度参数、关键字可变长度参数。
其中位置参数仅仅通过传入时的顺序来确定调用,并且如果没有任何默认参数的话,传入的参数数目必须和函数定义一致。python支持默认参数,将一些常用的参数初始化而避免重复写相同的参数。这两个都是常规函数的参数形式。另外一种非常规的参数就是参数的数目不一定,即可变长度的参数,这些参数可以是不带关键字的元组,或者是带关键字的字典。
python中,参数传入规则可以表示如下:
def myFunc([formal_args,] *args_tuple,**kw_dict)
其中formal_args表示位置参数,包含可能有的默认参数,而*args_tuple则表示可变长度的非关键字参数,**kw_dict表示可变长度的关键字参数。
匿名函数 lambada
匿名函数是python中对单行语句的特殊用法,用于简化一些简单函数。如
fun add(x,y):
return x+y
可以用lambada x,y:x+y来表示,lambada之后,冒号之前的内容表示需要传入的参数,冒号之后的内容表示执行的单行语句。
lambada可以被存储,以便之后调用,如 add = lambada x,y:x+y,可以调用为add(x,y)。
装饰器 decorator
由于函数的本质是一个引用,所以对函数名可以进行传入、传出等和一般变量一样的操作,通过这个特性,可以设计一些装饰器,用以对函数做重复的通用操作。python中对装饰器有额外的语法支持:@,在函数的上方加入@decorator语句,即表示对该函数进行装饰调用。一个函数可以被多个装饰器装饰。装饰器的本质是函数,所以装饰器也支持参数传入。一个装饰器的例子:
def log(func):
def wrappedFunc():
print ’The %s() is called!’ % func.__name__
return fun()
return wrappedFunc
通过@log的方式使用装饰器:
@log
def func():
pass
上述语句的含义是func=log(func),此时,func的内容会调用装饰器log,log函数最终会返回wrappedFunc函数对象,但是并没有调用该函数,当func()执行时真正调用wrappedFunc(),该函数首先打印一行调用日志,然后返回func()函数的调用,这里返回的调用,并不是func函数对象,所以在这里会真正执行func函数的实体。从而实现“装饰”的过程。
闭包 closure
在python2.1之后,作用域规则变为静态作用域,内部函数可以正常引用外部的变量,这种内部函数引用外部作用域的变量的函数就被称为闭包:
def counter(start_at = 0):
count = start_at
def incr():
count += 1
return count[0]
return incr
可以看到闭包的形式装饰器很像,区别是闭包传入的不一定是一个函数,只是对外部变量的引用,内部函数中也不一定需要执行传入的函数,可能做一些其他的工作。
闭包和装饰器的一个例子:
from time import time
def logged(when):
def log(f,*args,**kargs):
print ‘’’Called:
function: %s
args: %r
kargs: %r’’’ %(f,args,kargs)
def pre_logged(f):
def wrapped(*args,**kargs):
log(f,*args,**kargs)
return f(*args,**kargs)
return wrapped
def post_logged(f):
def wrapped(*args,**args):
now = time()
try:
return f(*args,**kargs)
finally:
log(f,*args,**kargs)
print “time delta: %s “ % (time()-now)
return wrapped
try:
return {‘pre’: pre_logged,’post’:post_logged}[when]
except KeyError,e:
raise ValueError(e),’must be “pre” or “post” '
@logged(“post")
def hello(name):
print “Hello,”,name
hello(“word!")
偏函数
通过引用functools 模块中的partial 可以引入偏函数,偏函数的概念就是对原本需要多个参数的函数确定其中一个参数而称为另一个函数:
from operator import add
from functools import partial
add1 = partial(add,1)
此时,add1(5) 就表示add(1,5)
作用域和globa语句
python搜索一个标示符的时候,会先从局部变量开始搜索,如果没有找到,则从上一级搜索,直到没有找到而返回NameError
通过globa关键字,可以将变量推出局部作用域,不通过上述查找过程,而直接引用一个已命名的全局变量。
生成器和yield语句
生成器用yield语句生成迭代器,每个yield语句顺序地使其内容在next()方法中被调用:
def simpleGen():
yield 1
yield 2
此时,simpleGen()就是一个迭代器,simpleGen().next() = 1,simpleGen().next() = 2
除了next()方法获得下一个值,还可以通过send方法来讲值送给生成器:
def counter(start_at = 0):
count = start_at
while True:
val = (yield count)
if val is not None:
count = val
else :
count += 1
count = count(5)
count.next() = 5;count.next() = 6;count.send(9) = 9; count.next() = 10; count.close() ; count.next(): StopIteration
Python学习_08_函数式编程的更多相关文章
- Python学习--08函数式编程
函数式编程的一个特点就是,允许把函数本身作为参数传入另一个函数,还允许返回一个函数. 高阶函数 Python支持高阶函数(Higher-order function). 什么是高阶函数呢?把函数作为参 ...
- Python学习:函数式编程(lambda, map() ,reduce() ,filter())
1. lambda: Python 支持用lambda对简单的功能定义“行内函数” 2.map() : 3.reduce() : 4.filter() : map() ,reduce() , filt ...
- Python进阶:函数式编程实例(附代码)
Python进阶:函数式编程实例(附代码) 上篇文章"几个小例子告诉你, 一行Python代码能干哪些事 -- 知乎专栏"中用到了一些列表解析.生成器.map.filter.lam ...
- Python Decorator 和函数式编程
看到一篇翻译不错的文章,原文链接: Python Decorator 和函数式编程
- Python基础:函数式编程
一.概述 Python是一门多范式的编程语言,它同时支持过程式.面向对象和函数式的编程范式.因此,在Python中提供了很多符合 函数式编程 风格的特性和工具. 以下是对 Python中的函数式编程 ...
- python学习_数据处理编程实例(二)
在上一节python学习_数据处理编程实例(二)的基础上数据发生了变化,文件中除了学生的成绩外,新增了学生姓名和出生年月的信息,因此将要成变成:分别根据姓名输出每个学生的无重复的前三个最好成绩和出生年 ...
- 可爱的 Python : Python中的函数式编程,第三部分
英文原文:Charming Python: Functional programming in Python, Part 3,翻译:开源中国 摘要: 作者David Mertz在其文章<可爱的 ...
- Python进阶之函数式编程(把函数作为参数)
什么是函数式编程? 什么是函数式编程? 函数:function 函数式:functional,一种编程范式 函数式编程是一种抽象计算的编程模式 函数≠函数式,比如:计算≠计算机 在计算机当中,计算机硬 ...
- Python之面向对象函数式编程
Python之面向对象函数式编程 函数式编程的根本就是用 def 去模拟数学式的编程逻辑. 类似与 y = 2*x + 1 ,当x = 3 时,函数的结果y就得7. def test(x): retu ...
随机推荐
- phpcms网站搬家 至 服务器 完整并且详细过程
上传服务器空间后,才会通过搜索域名进行网页访问. 上传的过程肯定会有很多东西要修改,例如数据库怎么上传.路径怎么修改等..... 这就让大家看下,自己不断尝试后的完整搬家步骤!!! 一.上传服务器 ( ...
- mac下出现xcrun: error导致git、svn无法使用的解决办法
现象:xcrun: error: invalid active developer path (/Library/Developer/CommandLineTools), missing xcrun ...
- linux系统下安装配置解压版的MySQL数据库
一.解压文件到当前目录 命令:tar -zxvf mysql....tar.gz 二.移动解压完成的文件夹到目标目录并更名mysql 命令:mv mysql-版本号 /usr/local/mysql ...
- Jenkins+tomcat+jdk setup
Jenkins download: http://jenkins-ci.org/ jdk version:jdk-7u45-linux-x64.tar.gz tomcat version:apache ...
- freemarker四种变量
freemarker四种变量 1.简单介绍说明 (1)数据模型中的变量:root中的变量 (2)模板中的变量:使用<#assign>定义的变量 (3)局部变量:在指令中的变量 (4)循环变 ...
- svn 批量加入没有加入版本号控制的文件命令
svn st | awk '{if($1=="?"){print$2}}'|xargs svn add
- Being a Good Boy in Spring Festival(杭电1850)(尼姆博弈)
Being a Good Boy in Spring Festival Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32 ...
- Python:list 和 array的对比以及转换时的注意事项
Python:list 和 array的对比以及转换时的注意事项 zoerywzhou@163.com http://www.cnblogs.com/swje/ 作者:Zhouwan 2017-6-4 ...
- 工作随笔——spring异步处理@Async使用笔记
@Async使用笔记 必须是public方法 必须是非static方法 方法调用的实例必须由spring创建和管理 代码示例如下: // 创建Foo类@Component class Foo { @A ...
- ESL翻译:Linear Methods for Regression
chapter 3: Linear Methods for Regression 第3章:回归的线性方法 3.1 Introduction A linear regression model assu ...