设为首页 收藏本站
查看: 1043|回复: 0

[经验分享] Python 装饰器的总结(二)

[复制链接]

尚未签到

发表于 2018-8-11 11:22:35 | 显示全部楼层 |阅读模式
  接着上一篇
  
  7,函数是Python世界中的一级类对象
  在Python里函数和其他东西一样都是对象
123456789#coding:UTF8print issubclass(int, object) Truedef foo():     passprint foo.__class__ # 1<type 'function'>print issubclass(foo.__class__, object)True  函数在Python里就是对象,和其他一样,在Python里,函数只是一些普通的值而已,也就是说把函数像参数一样传递给其他的函数或者从函数里返回函数,如:
1234567891011#coding:UTF8def add(x, y):     return x + ydef sub(x, y):     return x - ydef apply(func, x, y): # 1     return func(x, y) # 2print apply(add, 2, 1) # 33print apply(sub, 2, 1)1  在#1处看到函数准备接收一个函数的变量,只是一个普通的变量而已,和其他变量一样,在#2处调用传进来的函数:"()代表这调用函数的操作并且调用变量包含额值.在#3处,能看到传递函数并没有特殊的用法".函数的名称只是跟其他变量一样的标识符而已
  Python把频繁要用的操作变成函数作为参数进行使用,向通过传递一个函数给内置排序函数的key参数 从而 来自定义排序规则
123456789101112#coding:UTF8def outer():     def inner():         print "Inside inner"     return inner # 1  foo = outer() #2print foo<function inner at 0x000000000269C048> foo()Inside inner  在#1处恰好是函数标识符的变量inner作为返回值返回出来 "把函数inner返回出来,否则它根本不可能会被调用到" 每次函数outer呗调用,函数inner都会被重新定义,如果它不被当做变量返回额话,每次执行过后将不复存在
  在#2处捕获返回值--函数inner,将它存在一个新的变量foo里.当对foo进行求值,确定包含函数inner,而且能够对它进行调用
  8,闭包
1234567891011#coding:UTF8 def outer():     x = 1     def inner():         print x # 1     return innerfoo = outer()print foo.func_closure (<cell at 0x00000000026861F8: int object at 0x0000000001E279A8>,)  x是outer里的一个局部变量,当函数inner在#1处打印x时,Python解释器会在inner内部查找相应的变量,事实也查不到,接着会到封闭作用域里查找,并且找到匹配
  从变量的生存周期来看,变量x是函数outer的一个本地变量,意味着只有当函数outer正在运行时才会存在,根据Python运行模式,无法再函数outer返回之后继续调用函数inner,在函数inner调用时,变量x早已不复存在,可能会发生一个运行时的错误
  但返回的函数inner可以继续工作,Python支持一个叫做函数闭包的特性,嵌套定义在非全局作用域里的函数能够记住它在被定义的时候它所处的封闭命名空间,这能够通过查看函数的func_closure属性得出结论,这个属性里面包含封闭作用域里面的值(只会包含被捕捉到的值,比如x,如果在outer里面还定义了其他的值,封闭作用域里面是不会有的)
  每次函数outer被调用的时候,函数inner都会被重新定义。现在变量x的值不会变化,所以每次返回的函数inner会是同样的逻辑
  稍微改动下:
12345678910111213#coding:UTF8 def outer(x):     def inner():         print x # 1     return innerprint1 = outer(1)print2 = outer(2) print1()1print2()2  从中可以看到闭包--被函数记住的封闭作用域--能够被用来创建自定义的函数,本质上是一个硬编码的参数.事实上并不是传递参数1或者2给函数inner,实际上是创建了能够打印各种数字的各种自定义版本
  闭包单独拿出来就是一个非常强大的功能,在某些方面:outer像是给inner服务器的构造器,x像是一个私有变量
  9,装饰器
  装饰器其实就是一个闭包,把一个函数当做参数然后返回一个替代版参数
123456789101112131415#coding:UTF8 def outer(func):     def inner():         print "before func"         ret = func() # 1         return ret + 1     return innerdef foo():     return 1decorated = outer(foo) # 2print decorated() before func2  定义了一个函数outer,只有一个func参数,在其定义了嵌套的函数inner,inner会打印一串字符串,然后调用func,在#1得到返回值,在outer每次调用时func值可能会不一样,但不管怎用,都会调用它,最后,inner返回func()+1的值,通过调用在#2处存储decorated里的函数能够看到被打印出来的字符串以及返回值2,而不是期望中调用函数foo得到的返回值1。
  可以认为变量decorated是函数foo的一个装饰版本,一个加强版本。事实上如果打算写一个有用的装饰器的话,可能会想愿意用装饰版本完全取代原先的函数foo,这样总是会得到我们的”加强版“foo。想要达到这个效果,完全不需要学习新的语法,简单地赋值给变量foo就行了:
1foo = outer(foo)  现在,任何怎么调用都不会牵扯到原先的函数foo,都会得到新的装饰版本的foo,现在还是来写一个有用的装饰器
12345678910111213141516#coding:UTF8 import timedef bar():    time.sleep(2)    print('in the bar')def test2(func):    print(func)    return func # print(test2(bar))bar=test2(bar)bar()  #run bar <function bar at 0x00000000026BCF98>in the bar  10. 使用 @ 标识符将装饰器应用到函数和利用*args and **kwargs
  Python2.4支持使用标识符@将装饰器应用在函数上,只需要在函数的定义前加上@和装饰器的名称。在上一节的例子里我们是将原本的方法用装饰后的方法代替:
1bar=test2(bar)  这种方式能够在任何时候对任意方法进行包装。但是如果自定义一个方法,可以使用@进行装饰:
  #coding:UTF8
  import time
  def test2(func):
  print(func)
  return func
  @test2
  def bar():
  time.sleep(2)
  print('in the bar')
  bar()  #run bar
  _______________________________________________________________________________
  #coding:UTF8
  import time
  def timer(func): #timer(test1)  func=test1
  def deco(*args,**kwargs):
  start_time=time.time()
  func(*args,**kwargs)   #run test1()
  stop_time = time.time()
  print("the func run time  is %s" %(stop_time-start_time))
  return deco
  @timer  #test1=timer(test1)
  def test1():
  time.sleep(1)
  print('in the test1')
  @timer # test2 = timer(test2)  = deco  test2(name) =deco(name)
  def test2(name,age):
  print("test2:",name,age)
  test1()
  test2("Tom",22)
  in the test1
  the func run time  is 1.05200004578
  ('test2:', 'Tom', 22)
  the func run time  is 0.0
  _______________________________________________________________________________
  下面贡献一个高级版的装饰器:
  #coding:utf8
  import time
  user,passwd = 'hbert','abc'
  def auth(auth_type):
  print("auth func:",auth_type)
  def outer_wrapper(func):
  def wrapper(*args, **kwargs):
  #print("wrapper func args:", *args, **kwargs)
  if auth_type == "local":
  username = raw_input("Username:").strip()
  password = raw_input("Password:").strip()
  if user == username and passwd == password:
  print("\033[32;1mUser has passed authentication\033[0m")
  res = func(*args, **kwargs)  # from home
  print("---after authenticaion ")
  return res
  else:
  exit("\033[31;1mInvalid username or password\033[0m")
  elif auth_type == "ldap":
  print("搞毛线ldap,不会。。。。")
  return wrapper
  return outer_wrapper
  def index():
  print("welcome to index page")
  @auth(auth_type="local") # home = wrapper()
  def home():
  print("welcome to home  page")
  return "from home"
  @auth(auth_type="ldap")
  def bbs():
  print("welcome to bbs  page")
  index()
  print(home()) #wrapper()
  bbs()
  

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-550083-1-1.html 上篇帖子: python内置函数(上篇) 下篇帖子: Python 装饰器的总结(一)
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

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