python闭包和装饰器

简介: python闭包和装饰器

一、闭包

1.简单演示

条件

在函数嵌套(函数里面在定义函数)的前提下

内部函数使用了外部函数的变量(还包括外部函数的参数)

外部函数返回了内部函数

def outer():
    n = 1
    #内部定义
    def inner():
        #使用外部变量
        print(n)
    #返回内部函数    
    return inner
ret = outer()
print(ret)

def Person(name):
    def say(msg):
        print(name+"Say:  "+msg)
    return say
tom = Person('tom')
jack = Person('jack')
tom('你好')
jack('早上好')

2.内函数修改外函数变量问题

真常来说内部函数只有对外部函数访问的权利

def outer():
    n = 1
    def inner():
        n += 10
        print(n)
    print(n)
    return inner
func = outer()
func()

需要修饰符

def outer():
    n = 1
    def inner():
        nonlocal n
        n += 10
        print(n)
    print(n)
    return inner
func = outer()
func()

需要用nolocal


二、装饰器

1.基础使用

特点:

不修改已有函数的源代码

不修改已有函数的调用方法

给已有函数增加额外的功能

他的本质就是一个闭包函数

我们编辑一个计算函数用时功能的程序

import time
def for1():
    for i in range(10000000):
        i +=1
    print(i)
def start(func):
    def inner():
        s = time.time()
        func()
        e = time.time()
        print("用时:  ",e-s)
    return inner
func = start(for1)
func()

用装饰器来写

import time
def start(func):
    def inner():
        s = time.time()
        func()
        e = time.time()
        print("用时:  ",e-s)
    return inner
@start
def for1():
    for i in range(10000000):
        i +=1
    print(i)
for1()

由此可见装饰器就是一种语法糖

注意:装饰器这里一定要有参数

2.通用装饰器

参数问题

import time
def start(func):
    def inner():
        s = time.time()
        func()
        e = time.time()
        print("用时:  ",e-s)
    return inner
@start
def for1(a):
    for i in range(a):
        i +=1
    print(i)
for1(10000000000000000)

总结一下可能有的四种状态

#无参数,无返回值
def outer1(func):
    def inner():
        print("装饰1.。。")
        func()
        print("装饰2.。。")
    return inner
@outer1
def show1():
    print("show1....")
show1()
print("________________________")
# 有参数,无返回值
def outer2(func):
    def inner(msg):
        print("装饰1.。。")
        func(msg)
        print("装饰2.。。")
    return inner
@outer2
def show2(msg):
    print("show2....",msg)
show2("python")
print("________________________")
# 无参数,有返回值
def outer3(func):
    def inner():
        print("装饰1.。。")
        ret = func()
        print("装饰2.。。")
        return ret
    return inner
@outer3
def show3():
    return "shoe3....."
print(show3()+"Nihao")
print("________________________")
# 有参数,有返回值
def outer4(func):
    def inner(msg):
        print("装饰1.。。")
        ret = func(msg)
        print("装饰2.。。")
        return ret
    return inner
@outer4
def show4(msg):
    return "shoe3....." +msg
print(show4("python4    ")+"Nihao")

我们需要一个通用的修饰器

def outer(func):
    def inner(*args,**kwargs):
        print("装饰1.。。")
        ret = func(*args,**kwargs)
        print("装饰2.。。")
        return ret
    return inner
#无参数,无返回值
@outer
def show1():
    print("show1....")
show1()
print("________________________")
# 有参数,无返回值
@outer
def show2(msg):
    print("show2....",msg)
show2("python")
print("________________________")
# 无参数,有返回值
@outer
def show3():
    return "shoe3....."
print(show3()+"Nihao")
print("________________________")
# 有参数,有返回值
@outer
def show4(msg):
    return "shoe3....." +msg
print(show4("python4    ")+"Nihao")

3.多个修饰器引用

def wrapper_div(func):
    def inner(*args,**kwargs):
        return "<div>" + func(*args,**kwargs) + '</div>'
    return inner
def wrapper_p(func):
    def inner(*args,**kwargs):
        return "<p>" + func(*args,**kwargs) + '</p>'
    return inner
@wrapper_p
@wrapper_div
def show():
    return "show inner "
print(show())

执行流程

4.装饰器传参

需要多层函数嵌套来完成

def set_a(msg):
    def set_f(func):
        def inner():
            print("装饰内容: "+msg)
            func()
        return inner
    return set_f
@set_a("你好")
def show():
    print("show....")
show()


三、类装饰器

1.callable()和__call__

def show():
    print("show")
#这个函数用来测试参数是否是可以调用对象
print(callable(show))
print(callable(1))
print(callable("aa"))
class P():
    #当一个类中实现了__call__函数
    #那么这个类的实例对象就变成了可调用对象,也就是说,实例对象后面可以加()
    def __call__(self, *args, **kwargs):
        print("Call run")
tom = P()
print(callable(tom))
tom()

其实也就是说,类本来是不可调用的对象,但是装饰器需要是可调用对象,所以只需要了解这点我们就可以使用了

2.类装饰器的简单使用

class Wra():
    def __init__(self,func):
        self.func = func
    def __call__(self, *args, **kwargs):
        print("装饰")
        self.func()
@Wra
def show():
    print("show...")
show()


结语

这是最后几篇了,接下来更新Linux云计算方面的博客,大家👍啊!

相关文章
|
3天前
|
存储 缓存 算法
Python闭包|你应该知道的常见用例(下)
Python闭包|你应该知道的常见用例(下)
11 1
Python闭包|你应该知道的常见用例(下)
|
7天前
|
自然语言处理 小程序 测试技术
Python闭包|你应该知道的常见用例(上)
Python闭包|你应该知道的常见用例(上)
14 3
Python闭包|你应该知道的常见用例(上)
|
2天前
|
设计模式 缓存 监控
Python中的装饰器:代码的魔法增强剂
在Python编程中,装饰器是一种强大而灵活的工具,它允许程序员在不修改函数或方法源代码的情况下增加额外的功能。本文将探讨装饰器的定义、工作原理以及如何通过自定义和标准库中的装饰器来优化代码结构和提高开发效率。通过实例演示,我们将深入了解装饰器的应用,包括日志记录、性能测量、事务处理等常见场景。此外,我们还将讨论装饰器的高级用法,如带参数的装饰器和类装饰器,为读者提供全面的装饰器使用指南。
|
2天前
|
Python
深入浅出Python装饰器
【10月更文挑战第34天】在编程的世界里,我们常常需要扩展函数的功能,但又不想修改其源代码。Python装饰器的引入,就像是给函数穿上了一件魔法斗篷,让这一切变得可能。本文将带你领略装饰器的魔力,从基础概念到实际应用,一起探索这个强大的工具如何简化我们的代码并增加程序的可读性。
|
2天前
|
设计模式 程序员 数据处理
编程之旅:探索Python中的装饰器
【10月更文挑战第34天】在编程的海洋中,Python这艘航船以其简洁优雅著称。其中,装饰器作为一项高级特性,如同船上的风帆,让代码更加灵活和强大。本文将带你领略装饰器的奥秘,从基础概念到实际应用,一起感受编程之美。
|
3天前
|
缓存 开发者 Python
探索Python中的装饰器:简化和增强你的代码
【10月更文挑战第32天】 在编程的世界中,简洁和效率是永恒的追求。Python提供了一种强大工具——装饰器,它允许我们以声明式的方式修改函数的行为。本文将深入探讨装饰器的概念、用法及其在实际应用中的优势。通过实际代码示例,我们不仅理解装饰器的工作方式,还能学会如何自定义装饰器来满足特定需求。无论你是初学者还是有经验的开发者,这篇文章都将为你揭示装饰器的神秘面纱,并展示如何利用它们简化和增强你的代码库。
|
4天前
|
Python
探索Python装饰器:从入门到实践
【10月更文挑战第32天】在编程世界中,装饰器是一种特殊的函数,它允许我们在不改变原有函数代码的情况下,增加额外的功能。本文将通过简单易懂的语言和实际案例,带你了解Python中装饰器的基础知识、应用以及如何自定义装饰器,让你的代码更加灵活和强大。
11 2
|
6天前
|
设计模式 缓存 测试技术
Python中的装饰器:功能增强与代码复用的艺术####
本文将深入探讨Python中装饰器的概念、用途及实现方式,通过实例演示其如何为函数或方法添加新功能而不影响原有代码结构,从而提升代码的可读性和可维护性。我们将从基础定义出发,逐步深入到高级应用,揭示装饰器在提高代码复用性方面的强大能力。 ####
|
4天前
|
监控 Python
探索Python中的装饰器:从入门到实践
【10月更文挑战第31天】在Python的世界里,装饰器是那些隐藏在幕后的魔法师,它们拥有着改变函数行为的能力。本文将带你走进装饰器的世界,从基础概念到实际应用,一步步揭开它的神秘面纱。你将学会如何用几行代码增强你的函数功能,以及如何避免常见的陷阱。让我们一起来发现装饰器的魔力吧!
|
4天前
|
开发框架 开发者 Python
探索Python中的装饰器:技术感悟与实践
【10月更文挑战第31天】 在编程世界中,装饰器是Python中一种强大的工具,它允许我们在不修改函数代码的情况下增强函数的功能。本文将通过浅显易懂的方式,带你了解装饰器的概念、实现原理及其在实际开发中的应用。我们将一起探索如何利用装饰器简化代码、提高可读性和复用性,同时也会分享一些个人的技术感悟,帮助你更好地掌握这项技术。
18 2
下一篇
无影云桌面