Python必备基础之闭包和装饰器知识总结

一、闭包

1.1 三要素

  • 必须有一个内嵌函数
  • 内嵌函数必须引用外部函数中变量
  • 外部函数返回值必须是内嵌函数

1.2 语法

# 语法
def 外部函数名(参数):
	外部变量
	def 内部函数名(参数):
		使用外部变量
	return 内部函数名
# 调用
变量 = 外部函数名(参数)
变量(参数)

举个例子

def func01(): # 外部函数
	a = 1 # 外部变量
    print('外部变量:',a)
    def func02(num): #内部函数
    	print("调用内部函数后:",num + a). # 调用外部变量
# 调用
func01()
# func02()

像这样,我们先把func02注释掉,直接调用func01是可以调用成功的,完全没问题

但是我们再调用func02呢?一定会报错,为什么?这就涉及到一个知识点:
函数内部的属性,都是有生命周期的,都是在函数执行期间

简单来说就是func02存在于func01函数体内,func01调用执行完
它里面的代码就执行不了,如果想让它存活执行下去,就要return出去
再找一个变量接收,那么这样,不管你函数里怎么样,我就可以从内部使用外部的变量

所以调用这里不能像上面那么写:

# 调用
text = func01()
text(3) # 3为参数

这样才算整整意义上的闭包

1.3 优点

内部函数可以使用外部变量

1.4 缺点

外部变量一直存在于内存中,不会在调用结束后释放,占用内存

1.5 作用

实现python装饰器

二、装饰器 Decorator

2.1 定义

在不改变原函数的调用以及内部代码情况下,为其添加新功能的函数

这个常见的装饰器就是你拿到别人的第三方API,假如API接口不允许你修改
但是你觉得他写的特别low,还需要添加某些功能,那我们就需要使用装饰器

2.2 语法

def 函数装饰器名称(func):
    def wrapper(*args, **kwargs):
        需要添加的新功能
        return func(*args, **kwargs)
    return wrapper
原函数 = 内嵌函数
@函数装饰器名称
def 原函数名称(参数):
	函数体
原函数(参数)

2.3 本质

使用“@函数装饰器名称”修饰原函数,等同于创建与原函数名称相同的变量,关联内嵌函数;故调用原函数时执行内嵌函数。
原函数名称 = 函数装饰器名称(原函数名称)

2.4 装饰器链

一个函数可以被多个装饰器修饰,执行顺序为从近到远。

接下来我们写一个装饰器的小案例,来更加清楚一下装饰的整个工作流程
故事情境是这样的,主角是男人和女人,假设男人女人都是可以上班的,但是呢有不同

男人只能是好好上班,不能生娃;女人可以好好上班,也可以生娃

当然我们别反驳啊,是有的国家的男的也有生娃的技术,但是我们这里就是按照我们设定好的来

那当我们调用 man() 的时候,打印 好好上班,你不能生娃
调用 woman() 的时候,打印 好好上班,你可以生娃

def man():
	print("好好上班")
def woman():
	print("好好上班")

man()
woman()

那我们紧接着构建装饰器,装饰器名字无所谓,想怎么定义就怎么定义

# 装饰器函数带参数
def arg_func(sex):
	def func1(b_func):
		def func2():
			if sex == 'man':
				print("你不可以生娃")
			if sex == 'woman':
				print("你可以生娃")
			return b_func()
		return func2
	return func1
@arg_func(sex='man')
def man():
	print("好好上班")
@arg_func(sex='woman')
def woman():
	print("好好上班")

man()
woman()

这个生成器大概的过程就是:

arg_func(sex='man'/'woman')()() > func1
func1() > func
func()  > print("你不可以生娃") or print("你可以生娃") > b_func
# 然后判断sex的值,最后return出去,拿到结果

我们看这个生成器啊,因为它这个函数这里**def arg_func(sex)😗*这里是默认接收一个函数名作为参数进来,但是现在参数有了,但是函数名不见了,怎么办,我们只能是去在这个函数里再次写一个函数,再传入函数名

这个就是相当于只要是有传递参数的话,就要在写一个函数套进去,因为你还有一个函数名要进行传参

接下来我们看一下被装饰的函数带参数

def func1(func):
    def func2(x, y):
        print(x, y)
        x += 5
        y += 5
        return func(x, y)
    return func2
@func1
def num_sum(a, b):
    print(a + b)
num_sum(1, 2)

这种装饰器跟之前的相比,直观的感受来说代码减少,更加精简,所以我们常用的也是这个较多

它总体来说流程变化不大,就是对于传参的形式进行了变化,即采用函数最内部传参

到此这篇关于Python必备基础之闭包和装饰器知识总结的文章就介绍到这了,更多相关Python闭包和装饰器内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • 详解Python中的装饰器、闭包和functools的教程

    装饰器(Decorators) 装饰器是这样一种设计模式:如果一个类希望添加其他类的一些功能,而不希望通过继承或是直接修改源代码实现,那么可以使用装饰器模式.简单来说Python中的装饰器就是指某些函数或其他可调用对象,以函数或类作为可选输入参数,然后返回函数或类的形式.通过这个在Python2.6版本中被新加入的特性可以用来实现装饰器设计模式. 顺便提一句,在继续阅读之前,如果你对Python中的闭包(Closure)概念不清楚,请查看本文结尾后的附录,如果没有闭包的相关概念,很难恰当的理解P

  • 简析Python的闭包和装饰器

    什么是装饰器? 装饰器(Decorator)相对简单,咱们先介绍它:"装饰器的功能是将被装饰的函数当作参数传递给与装饰器对应的函数(名称相同的函数),并返回包装后的被装饰的函数",听起来有点绕,没关系,直接看示意图,其中 a 为与装饰器 @a 对应的函数, b 为装饰器修饰的函数,装饰器@a的作用是: 简而言之:@a 就是将 b 传递给 a(),并返回新的 b = a(b) 栗子: 上面使用@dobi来表示装饰器,其等同于:qinfeng = dobi(qinfeng) 因此装饰器本质

  • Python的几个高级语法概念浅析(lambda表达式闭包装饰器)

    1. 匿名函数 匿名函数(anonymous function)是指未与任何标识符绑定的函数,多用在functional programming languages领域,典型应用场合: 1) 作为参数传给高阶函数(higher-order function ),如python中的built-in函数filter/map/reduce都是典型的高阶函数 2) 作为高阶函数的返回值(虽然此处的"值"实际上是个函数对象) 与命名函数(named function)相比,若函数只被调用1次或有

  • Python高级特性之闭包与装饰器实例详解

    本文实例讲述了Python高级特性之闭包与装饰器.分享给大家供大家参考,具体如下: 闭包 1.函数参数: (1)函数名存放的是函数的地址 (2)函数名()存放的是函数内的代码 (3)函数名只是函数代码空间的引用,当函数名赋值给一个对象的时候,就是引用传递 def func01(): print("func01 is show") test = func01 print(func01) print(test) test() 结果: 2.闭包: (1)内层函数可以访问外层函数变量 (2)闭

  • Python闭包与装饰器原理及实例解析

    一.闭包 闭包相当于函数中,嵌套另一个函数,并返回.代码如下: def func(name): # 定义外层函数 def inner_func(age): # 内层函数 print('name: ', name, ', age: ', age) return inner_func # 注意此处要返回,才能体现闭包 bb = func('jayson') # 将字符串传给func函数,返回inner_func并赋值给变量 bb(28) # 通过变量调用func函数,传入参数,从而完成闭包 >>

  • 详解 Python中LEGB和闭包及装饰器

    详解 Python中LEGB和闭包及装饰器 LEGB L>E>G?B L:local函数内部作用域 E:enclosing函数内部与内嵌函数之间 G:global全局作用域 B:build-in内置作用域 python 闭包 1.Closure:内部函数中对enclosing作用域变量的引用 2.函数实质与属性 函数是一个对象 函数执行完成后内部变量回收 函数属性 函数返回值 passline = 60 def func(val): if val >= passline: print (

  • 详解python中的生成器、迭代器、闭包、装饰器

    迭代是访问集合元素的一种方式.迭代器是一个可以记住遍历的位置的对象.迭代器对象从集合的第一个元素开始访问,直到所有的元素被访问完结束.迭代器只能往前不会后退. 1|1可迭代对象 以直接作用于 for 循环的数据类型有以下几种: 一类是集合数据类型,如 list . tuple . dict . set . str 等: 一类是 generator ,包括生成器和带 yield 的generator function. 这些可以直接作用于 for 循环的对象统称为可迭代对象: Iterable .

  • Python闭包装饰器使用方法汇总

    闭包内容: 匿名函数:能够完成简单的功能,传递这个函数的引用,只有功能 普通函数:能够完成复杂的功能,传递这个函数的引用,只有功能 闭包:能够完成较为复杂的功能,传递这个闭包中的函数以及数据,因此传递是功能+数据 对象:能够完成最复杂的功能,传递很多数据+很多功能,因此传递的是数据+功能 ------------------- 对全局函数进行修改:在函数当中加global,在闭包中外边中的变量加nonlocal 闭包定义:有两个函数嵌套使用,里面的函数可以使用外面函数所传输的参数,最后可传递的是

  • 深入理解python中的闭包和装饰器

    python中的闭包从表现形式上定义(解释)为:如果在一个内部函数里,对在外部作用域(但不是在全局作用域)的变量进行引用,那么内部函数就被认为是闭包(closure). 以下说明主要针对 python2.7,其他版本可能存在差异. 也许直接看定义并不太能明白,下面我们先来看一下什么叫做内部函数: def wai_hanshu(canshu_1): def nei_hanshu(canshu_2): # 我在函数内部有定义了一个函数 return canshu_1*canshu_2 return

  • Python闭包和装饰器用法实例详解

    本文实例讲述了Python闭包和装饰器用法.分享给大家供大家参考,具体如下: Python的装饰器的英文名叫Decorator,作用是完成对一些模块的修饰.所谓修饰工作就是想给现有的模块加上一些小装饰(一些小功能,这些小功能可能好多模块都会用到),但又不让这个小装饰(小功能)侵入到原有的模块中的代码里去. 闭包 1.函数引用 #coding=utf-8 def test1(): print('This is test1!') #调用函数 test1() #引用函数 ret = test1 #打印

随机推荐