Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 483|回复: 0

python 装饰器问题及解决方式

[复制链接]
  • TA的每日心情
    奋斗
    2024-4-6 11:05
  • 签到天数: 748 天

    [LV.9]以坛为家II

    2034

    主题

    2092

    帖子

    70万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    705612
    发表于 2021-7-14 10:30:55 | 显示全部楼层 |阅读模式
     1 #统计函数运行时间
     2 import time
     3 #装饰器
     4 def timmer(func):
     5     def wrapper(*args,**kwargs):
     6         start_time=time.time()
     7         func()
     8         stop_time =time.time()
     9         print("the func time is %s"%(stop_time-start_time))
    10 
    11 #函数
    12 @timmer
    13 def test1():
    14     time.sleep(3)
    15     print('in the test1')
    16     
    17 test1()

    上面函数运行会出现报错:

    函数调用时出错;
    'NoneType' object is not callable
    此时将调用函数语句去掉后面的括号,改为 test1即可。
    装饰器:本质是函数(装饰其他函数)就是为其他函数添加附加功能 
    高阶函数+嵌套函数==》装饰器
    原则:不能修改被装饰函数的原代码
             不能修改被装饰函数的调用方式
    实现装饰器:
    1.函数即‘’变量‘’
    2.高阶函数: 满足两个条件之一
    a:把一个函数名当做实参传给另外一个函数(在不修改被装饰函数的源代码的情况下,为其添加功能)。
    b:返回值中包含函数名。(不修改函数的调用方式)
    实现高阶函数a
     1 import time
     2 def bar():
     3     time.sleep(3)
     4     print('in the bar')
     5 #装饰器,
     6 def test1(func):
     7     print(func)
     8     start_time=time.time()
     9     func()   #run bar
    10     stop_time = time.time()
    11     print('the func runtime is %s'%(stop_time-start_time))
    12     
    13 test1(bar)  #改变了函数原本的调用方式
    1 #装饰器中,使用函数作为变量,通过变量也能调用该函数。
    2 
    3 func=bar()
    4 
    5 func()
    6 
    7 #此处和bar()的执行结果是一样的

    刚开始写装饰器时,对函数调用方式有疑惑。上面代码最后的语句:test(bar)改变函数原本调用方式很不理解。

    经过断点调试和查阅,理解如下:
    原函数没有传参,bar()得到的是bar这个函数的执行结果,即等待三秒后执行输出。但是使用装饰器后,高阶函数test1(bar)函数的为原函数附加了一个功能,即输出函数的运行时间,但函数的调用变成将bar的内存地址赋给func,此处使用func()就是执行bar这个函数。
    严格意义来说,上面的代码并不符合装饰器的原则。
    实现高阶函数b:返回值是函数名。没有改变函数的调用方式。
     1 import time
     2 #函数的调用方式没有改变
     3 def bar():
     4     time.sleep(3)
     5     print('in the bar')
     6 #装饰器
     7 def test2(func):
     8     print(func)
     9     return func
    10 print('--->',test2(bar))#t=test2(bar)   print(t)  二者实现的效果是一样的
    11 t=test2(bar)
    12 t()#等同于bar()
    13 
    14 bar=test2(bar)
    15 bar()#等同于 run bar

        上面内容:把函数bar传给func,print(func),实质是打印下来bar的内存地址,return func 的返回值就是test2(bar)的运行结果。          print('--->',test2(bar))就是将test2(bar)的运行结果(即return func)的运行结果打印下来

    第11行中:test2(bar())和test2(bar)的区别?

    前者是将bar的运行结果传给test2,(即字符串类型),不符合高阶函数中返回值是函数名这一原则。

    后者将bar的内存地址返回给test2,之后执行。

     

    3.函数嵌套
    定义:在一个函数体内,用def声明一个新的函数。(注意与函数调用的区别)

     
    4.局部作用域和全局作用域的访问顺序

     1 #局部作用域和全局作用域的访问顺序
     2 x=0
     3 def test1():
     4     x=1
     5     print(x)
     6     def test2():
     7         x=2
     8         print(x)
     9         def test3():
    10             x=3
    11             print(x)
    12         test3()
    13     test2()
    14 test1()
    15 print(x)

    通过上面实现可以看出:

    若将test2()注释掉,则将不能访问进入test3()。函数外层输出的x的结果仍旧是0,因为它是全局变量。

     

     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
     
    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2024-5-19 09:34 , Processed in 0.072242 second(s), 29 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表