在Python语言中,可以在函数中定义函数. 这种在函数中嵌套定义的函数也叫内部函数.我们来看下面的代码:
上述代码中,定义了函数greet,在函数greet内部又定义了一个函数inner_func, 并调用该函数打印了一串字符.
我们可以看到,内部函数inner_func的定义和使用与普通函数基本相同.需要注意的是变量的作用域,在上述代码中,函数参数name对于全局函数greet是局部变量,对内部函数inner_func来说则是非局部变量.内部函数对于非局部变量的访问规则类似于标准的外部函数访问全局变量.
从这个例子我们还可以看到内部函数的一个作用,就是通过定义内部函数的方式将一些功能隐藏起来,防止外部直接调用.常见的场景是,在一个复杂逻辑的函数中,将一些小的任务定义成内部函数,然后由这个外层函数使用,这样可以使代码更为清晰,易于维护.这些内部函数只会在这个外层函数中使用,不能被其他函数或模块使用.
在Python语言中, 函数也是对象,它可以被创建、赋值给变量,或者作为函数的返回值.我们来看下面这个例子.
在上述代码中,在函数gen_greet内部定义了inner_func函数,并返回了一个inner_func函数对象.外部函数gen_greet返回了一个函数对象,所以像gen_greet这样的函数也叫工厂函数.
在内部函数inner_func中,使用了外部函数的传参greet_words(非局部变量),以及函数的参数name(局部变量),来打印一个字符串.
此时此刻呢,调用gen_greet("Hello")创建一个函数对象say_hello,紧接着调用say_hello("Mr. Zhang"),输出的结果为:Hello, Mr. Zhang!
同样的,调用gen_greet("Hi")创建一个函数对象say_hi,调用say_hello("Mr. Zhang"),输出的结果为:Hi,Tony!
我们可以发现,gen_greet返回的函数对象具有记忆功能,它能够把所需使用的非局部变量保存下来,用于后续被调用的时候使用.这种保存了非局部变量的函数对象被称作闭包(closure).
那么闭包是如何实现的呢?其实并不复杂,函数对象中有一个属性__closure__,它就是在创建函数对象时用来保存这些非局部变量的.
__closure__属性是一个元组或者None类型.在上述代码中,我们可以通过下面方式查看:
函数的嵌套所实现的功能大都可以通过定义类的方式来实现,而且类是更加面向对象的代码编写方式.
嵌套函数的一个主要用途是实现函数的装饰器.我们看下面的代码:
通过了解了嵌套函数和闭包的工作原理,我们在使用过程中就能够更加得心应手了.
嵌套函数在执行时(而不是在定义时)从父范围中查找变量.
编译函数主体,然后验证"自由"变量(未在函数本身中通过赋值定义),然后将其作为闭包单元绑定到函数,并且代码使用索引引用每个单元格.pet_function所以呢具有一个自由变量(cage),然后将其通过一个闭合单元引用,索引为0的闭合本身指向局部变量cage在get_petters功能.
当你实际调用该函数时,该闭包将用于在你调用该函数时查看cage周围作用域中的值.问题就今天这一节.在你调用函数时,该函数已经完成了对其结果的计算.将在在执行过程中的一些点局部变量分配各的,和字符串,但在功能的结束,包含了最后一个值.所以呢,当你调用每个动态返回的函数时,就会得到打印的值.get_petterscage'cow''dog''cat'cage'cat''cat'
解决方法是不依赖闭包.你可以改用部分函数,创建新的函数作用域或将变量绑定为关键字parameter的默认值.
部分函数示例,使用functools.partial():
from functools import partialdef pet_function(cage=None):
print "Mary pets the " + cage.animal + "."yield (animal, partial(gotimes, partial(pet_function, cage=cage)))
创建一个新的范围示例:
def scoped_cage(cage=None):
def pet_function():
print "Mary pets the " + cage.animal + "."
return pet_functionyield (animal, partial(gotimes, scoped_cage(cage)))
将变量绑定为关键字参数的默认值:
def pet_function(cage=cage):
print "Mary pets the " + cage.animal + "."yield (animal, partial(gotimes, pet_function))
无需scoped_cage在循环中定义函数,编译仅进行一次,而不是在循环的每次迭代中进行.
在前面已经多次提到函数这个概念,之所以没有解释什么是函数,是因为程序中的函数和数学中的函数差不多,如input()、range()等都是函数,这些都是Python的标准函数,直接使用就可以了.根据需要,用户也可以自定义函数.
函数的结构:
def 函数名(参数):
函数体
return 返回值
def f(x):
return(y)
下面给出完整的程序代码:
print(res)
运行结果:11
上例中的y是函数f(x)的返回值.并不是所有的函数都有参数和返回值.如下面的函数:
def func():
print('此为无参数传递、无返回值的函数')
func()
输出结果:此为无参数传递、无返回值的函数
可以看出,该函数func()无参数,故调用时不用赋给参数值.
def f(x,y):
return z
也可以通过直接给参数列表中的参数赋值的方法,为参数添加默认值,如果用户赋予参数值,则按照用户赋值执行,否则使用默认值.例如:
回调函数,又称函数回调,是将函数作为另一函数的参数.
例如:
def func(fun,m,n):
fun(m,n)
def f_add(m,n):
print('m+n=',m+n)
def f_mult(m,n):
print('m*n=',m*n)
输出结果:
在f_add(m,n)和f_mult(m,n)被定义前,func(fun,m,n)中的fun(m,n)就已经调用了这两个函数,即"先调用后定义",这也是回调函数的特点.
如果无法预知参数的个数,可以在参数前面加上*号,这种参数实际上对应元组类型.譬如,参会的人数事先不能确定,只能根据与会人员名单输入:
def func(*names):
print('今天与会人员有:')
for name in names:
print(name)
func('张小兵','陈晓梅','李大海','王长江')
运行后,输出结果为:
今天与会人员有:
张小兵
陈晓梅
李大海
王长江
参数为字典类型,需要在参数前面加上**号.
def func(**kwargs):
for i in kwargs:
print(i,kwargs[i])
func(a='a1',b='b1',c='c1')
输出结果为:
a a1
b b1
c c1
一个有趣的实例:
def func(x,y,z,*args,**kwargs):
print(x,y,z)
print(args)
print(kwargs)
func('a','b','c','Python','is easy',py='python',j='java',ph='php')
a b c # 前三个实参赋给前三个形参
('Python', 'is easy') # *args接收元组数据
{'py': 'python', 'j': 'java', 'ph': 'php'} # **kwargs接收字典数据
变量的作用域即变量的有效范围,可分为全局变量和局部变量.
局部变量
在函数中定义的变量就是局部变量,局部变量的作用域仅限于函数内部使用.
全局变量
在主程序中定义的变量就是全局变量,但在函数中用关键字global修饰的变量也可以当做全局变量来使用.
全局变量的作用域是整个程序,也就是说,全局变量可以在整个程序中可以访问.
下面通过实例去讨论:
程序1:
a=1 # a为全局变量
def a_add():
print('a的初值:',a) # 在函数中读取a的值
a_add() # 调用函数a_add()
a+=1 # 主程序语句,a增加1
print('a现在的值是:',a) # 主程序语句,读取a的值
运行结果:
a的初值: 1
这个结果和我们想象的一样,全局变量a既可以在主程序中读取,也可以在子程序(函数)中读取.
a=1
a+=1
print('a的初值:',a)
a_add()
print('a现在的值是:',a)
运行程序1时出现如下错误提示:
UnboundLocalError: local variable 'a' referenced before assignment
意思是:局部变量'a'在赋值之前被引用.
怎样解决这个问题?
def a_add(x):
x+=1
return x
a=a_add(a)
结果的确是正确的,但在函数a_add(x)中没有调用变量a(没有出现变量a).
def a_add(a):
return a
global a
虽然使用global可使变量使用起来非常方便,但也容易引起混淆,故在使用过程中还是谨慎为好.
递归,就是函数调用它自身.递归必须设置停止条件,否则函数将无法终止,形成死循环.
以计算阶乘为例:
def func(n):
if n==1:
return 1
else:
return n*func(n-1) #func( )调用func( )
嵌套,指在函数中调用另外的函数.这是程序中常见的一种结构,在此不再赘述.
匿名函数
Python中可以在参数前加上关键字lambda定义一个匿名函数,这样的函数一般都属于"一次性"的.
程序1:这是一个常规的函数定义和调用.
def f_add(x,y):
return x+y
f_add=lambda x,y:x+y
第一段:函数的定义
函数是指将一组语句的集合通过一个名字(函数名)封装起来,想要执行这个函数,只需要调用函数名即可
特性:
减少重复代码
使程序变得可扩展
使程序变得易维护
第二段:函数的参数
形参,调用时才会存在的值
实惨,实际存在的值
定义:当不输入参数值会有一个默认的值,默认参数要放到最后
定义: 正常情况下,给函数传参数要安装顺序,不想按顺序可以用关键参数,只需要指定参数名即可,(指定了参数名的就叫关键参数),但是要求是关键参数必须放在位置参数(以位置顺序确定对应的参数)之后
定义: 如你的函数在传入参数时不确定需要传入多少个参数,就可以使用非固定参数
# 通过元组形式传递
# 通过列表形式传递
# 字典形式(通过k,value的方式传递)
# 通过变量的方式传递
第三段:函数的返回值
作用:
返回函数执行结果,如果没有设置,默认返回None
终止函数运行,函数遇到return终止函数
第四段:变量的作用域
全局变量和局部变量
在函数中定义的变量叫局部变量,在程序中一开始定义的变量叫全局变量
全局变量作用域整个程序,局部变量作用域是定义该变量的函数
当全局变量与局部变量同名是,在定义局部变量的函数内,局部变量起作用,其他地方全局变量起作用
同级的局部变量不能互相调用
想要函数里边的变量设置成全局变量,可用global进行设置
第五段:特殊函数
定义: 嵌套函数顾名思义就是在函数里边再嵌套一层函数
提示 在嵌套函数里边调用变量是从里往外依次调用,意思就是如果需要调用的变量在当前层没有就会去外层去调用,依次内推
基于Lambda定义的函数格式为: lambda 参数:函数体
参数,支持任意参数.
匿名函数适用于简单的业务处理,可以快速并简单的创建函数.
# 与三元运算结合
定义:变量可以指向函数,函数的参数可以接收变量,那么一个函数就可以接收另一个函数作为参数,这种函数称之为高阶函数 只需要满足一下任意一个条件,即是高阶函数
接收一个或多个函数作为输入
return返回另一个函数
定义:一个函数可以调用其他函数,如果一个函数调用自己本身,这个函数就称为递归函数
递归实现过程是先一层一层的进,然后在一层一层的出来
必须有一个明确的条件结束,要不然就是一个死循环了
每次进入更深层次,问题规模都应该有所减少
递归执行效率不高,递归层次过多会导致站溢出
# 打印数字从1-100
定义:内层函数调用外层函数的变量,并且内存函数被返回到外边去了
闭包的意义:返回的函数对象,不仅仅是一个函数对象,在该函数外还包裹了一层作用域,这使得,该函数无论在何处调用,优先使用自己外层包裹的作用域
以上就是土嘎嘎小编为大家整理的python嵌套函数变量相关主题介绍,如果您觉得小编更新的文章只要能对粉丝们有用,就是我们最大的鼓励和动力,不要忘记讲本站分享给您身边的朋友哦!!