python函数嵌套、闭包与装饰器

简介:

一、函数知识

1、函数定义与调用

函数的定义形式:

1
2
3
4
5
6
7
def  '函数名' (参数 1 ,参数 1 ,参数n):
     函数功能体
     return   "结果"    #可选
#样例函数
def  hello(n):
     print ( "Hello %s"  % n)
     return  "hello() finish."

说明:

    函数名和变量的命名规则类似,这里不详述;参数可以没有,也可以有多个,默认参数等,参数的类型这里不讨论;可能有return返回计算结果,一个函数也可以没有return,如果没有return语句,则默认返回"None".

样例函数中,在交互下输入hello函数名 或print(hello)会出现类似 <function hello at 0x01FBDEF0>这可以理解成是hello函数在内存中的地址空间即函数自身; 而hello("world")则hello函数的调用,会出现"Hello world";print(hello("world")) 会出现 "Hello world"和 "hello finish." 如图:

wKioL1lbU1mT8hDCAAAdiGS3fD8005.png-wh_50



2、return 与print区别

在明白了函数的定义和调用以后,来说说print和return区别;

两者主要区别 :print是打印内容,可以出现在任何地方,而return只出现在函数中,用来返回函数运行计算的结果(传给其他函数),同时结束函数,return返回的结果需要print显示才能打印出来;理论上任何一个函数都有一个返回值,如果没有明确return 结果,函数默认返回None。

例子:

1
2
3
4
5
def  hi():
     print ( "I am hi." )
def  hello():
     return ( "I am hello." )
     print ( "I can't appear!" )

调用hi()    打印出 "I am hi."

调用hello()   没有任何显示,调用结果是返回字符串"I am hello." 不过没有打印出来,如果想hello()调用显示出来需要print(hello()) ,别外 print("I can't appear!")这一句是不会执行的,原由就是如上所说,在碰到return后函数就结束了;而如此同时print(hi())  则默认return "None"如下所示:

wKioL1lbT_LQiWMsAAAPLRdrqO4015.png


二、函数闭包

定义:在函数内部对嵌套层函数进行返回操作,如下:

1
2
3
4
def  hello():
     def  hi():
         print ( "hi in hello!" )
     return  hi                        #第4行

说明:我们调用hello()函数时,没有结果,print(hello()) 同样是返回hi函数本身

调用hello()() 才是类似单独调用定义的hi调用,结果如下:

wKiom1lbWfaC4EDNAAASKiBfyEE878.png

注意第4行,这里要是写成hi() 即hi函数的调用那么 ,hello()时就会出现 结果"hi in hello!"

hello()()就会出错啦~这里请注意,原因就是本文一开始就说明的函数与函数调用即returun返回了hi调用结果,返回了"hi in hello!",赶紧动手试试吧!


三、函数装饰器

定义:在不改变某一函数(outer)的情况下,增强另一个函数的功能(inner),这种在代码运行期间动态增加功能的方式,称之为“装饰器”(Decorator)。

1
2
3
4
5
6
7
8
9
def  outer(func):
     def  inner():
         print ( "inner is start!" )
         func()
         print ( "inner is finished!" )
     return  inner
  
def  hello():
     print ( "Hello World!" )


说明:outer就是一个类似闭包函数,不同是outer提供了一个接口(参数,这个参数是函数就是加强的功能)因此可以通过 以下方式使用outer函数:

1、传入hello函数(参数):

print("*" *10)
outer(hello)()

wKiom1lbYnvgPdmyAAARlSCO3Dg118.png

2、传入hello函数并赋给A,调用A()
print("*" *10)
A = outer(hello)
A()

wKiom1lbYozi6hI4AAAHeedNg6M071.png

3、赋给同名hello变量,覆盖原来的hello函数

类似 于hello = hello + outer,新的hello整合了hello ,outer的功能

print("*" *10)
hello = outer(hello)
hello()

wKiom1lbYp7CsojWAAAHe12hZPA568.png-wh_50

4、装饰器的写法

@outer

def hello():

    print("Hello World2!")    #这里为了区分上面的hello

等同于hello = outer(hello)

wKiom1lbYsOzblx0AAALQ6xM9Fc142.png

5、增强outer功能

@outer
def hi():
    print("Hi,I am hi!")
    
print("*" *10)
hi()

wKiom1lbYs_zI2UjAAAJ2vipRdA025.png


6、函数带参数的装饰器

以上的函数比较简单没有带任何参数,那么要是被装饰的函数带参数呢?

代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
def  outer(f):
     def  inner( * args, * * kargs):
         print ( "inner is start!" )
         f( * args, * * kargs)
         print ( "inner is finish!" )
     return  inner
 
     
@outer 
def  hi( * args, * * kargs):
     for  name  in  args:
         print ( "hi %s"  % name)
     print (kargs)
             
hi( "san" , "test" ,san = 18 )

运行以上代码结果如下:

968b34ac5e501509c7c563566bf20575.png-wh_


7、装饰器带参数用法

模拟需求:

以上的函数和装装饰器基础上要求在不改变函数功能的情况下,当函数执行时通过装饰器传参数,可控记录调用日志。即,当传递”true“时记录日志,什么也不传就不记录日志。


分析:在不改变函数情况下,改装饰器,以上的装装饰器只给函数(带参数的函数)扩充功能,能否直接在装饰器@outer后加参数传递呢?如果看到这里明白了装饰器的核心就会知道,以上例子中直接加参数是不行,@outer("参数")是行不通的 ,因为核心@outer 等同于func = outer(func)  所以直接@outer("参数") 是不可以的。实现的方法如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
def  loger(flag = "flash" ):
     def  outer(func):
         def  inner():
             print ( "inner is start!" )
             func()
             print ( "inner is finished!" )
             if  flag  = =  "true" :
                 print ( "%s 函数调用日志已经记录"  % func.__name__)
         return  inner
     return  outer
 
@loger ()
def  hello():
     print ( "Hello World!" )
 
@loger ( "true" )
def  hello2():
     print ( "Hello2 World!" )
 
hello()
print ( "****分割线****"  * 5 )
hello2()


运行结果如图:

be1b0127c874d54cfd0d745be73a0bb9.png-wh_

可以看到通过在原有的装饰器函数中加一层,这一层主要目的就是类似闭包功能用来向原有装饰器函数中传递变量(参数)。


以上是本人总结,如有错误之处欢迎指正!










本文转自 dyc2005 51CTO博客,原文链接:http://blog.51cto.com/dyc2005/1944505,如需转载请自行联系原作者
相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
目录
相关文章
|
8天前
|
Python
python函数的参数学习
学习Python函数参数涉及五个方面:1) 位置参数按顺序传递,如`func(1, 2, 3)`;2) 关键字参数通过名称传值,如`func(a=1, b=2, c=3)`;3) 默认参数设定默认值,如`func(a, b, c=0)`;4) 可变参数用*和**接收任意数量的位置和关键字参数,如`func(1, 2, 3, a=4, b=5, c=6)`;5) 参数组合结合不同类型的参数,如`func(1, 2, 3, a=4, b=5, c=6)`。
13 1
|
2天前
|
数据安全/隐私保护 Python
Python中的装饰器:提升代码可读性与灵活性
Python中的装饰器是一种强大的工具,可以在不改变函数原有逻辑的情况下,为函数添加额外的功能。本文将介绍装饰器的基本概念和用法,并通过实例演示如何利用装饰器提升代码的可读性和灵活性,使代码更加简洁、易于维护。
|
3天前
05-python之函数-函数的定义/函数的参数/函数返回值/函数说明文档/函数的嵌套使用/函数变量的作用域
05-python之函数-函数的定义/函数的参数/函数返回值/函数说明文档/函数的嵌套使用/函数变量的作用域
|
4天前
|
Python
python学习10-函数
python学习10-函数
|
4天前
|
Python
python学习4-内置函数range()、循环结构、循环控制语句、else语句、嵌套循环
python学习4-内置函数range()、循环结构、循环控制语句、else语句、嵌套循环
|
4天前
|
程序员 Python
Python中的装饰器:提升代码可读性与灵活性
在Python编程中,装饰器是一种强大的工具,可以在不修改原始代码的情况下,动态地添加功能。本文将深入探讨Python中装饰器的原理、用法和实际应用,以及如何利用装饰器提升代码的可读性和灵活性。
|
6天前
|
缓存 开发者 Python
深入探讨Python中的装饰器:提升代码可读性与灵活性
在Python编程中,装饰器是一种强大的工具,可以在不修改原始函数代码的情况下,对其行为进行扩展或修改。本文将深入探讨装饰器的原理和用法,以及如何利用装饰器提升代码的可读性和灵活性,为Python开发者提供更加优雅和高效的编程方式。
|
7天前
|
测试技术 开发者 Python
Python中的装饰器:优雅而强大的函数修饰工具
在Python编程中,装饰器是一种强大的工具,用于修改函数或方法的行为。本文将深入探讨Python中装饰器的概念、用法和实际应用,以及如何利用装饰器实现代码的优雅和高效。
|
1月前
|
缓存 API Python
Python中的装饰器:优雅而强大的函数增强工具
在Python编程中,装饰器是一种强大的工具,可以在不改变原函数代码的情况下,对函数进行增强和扩展。本文将介绍装饰器的基本概念、用法以及实际应用场景,帮助读者更好地理解并运用装饰器提升代码的可维护性和灵活性。
|
1月前
|
Python
Python用装饰器实现函数计时功能。
【2月更文挑战第11天】【2月更文挑战第29篇】Python用装饰器实现函数计时功能。

热门文章

最新文章