python 中的decorator

python 中decorator的作用就是一个包装的作用,所谓包装指在执行真正的函数之前或者之后,我们可以有一些额外的发挥余地。

decorator形式如下

 

def dec(arg1):
	print("dec",arg1)
	def func_replace(func):
		print("func_replace",func)
		
		def call_real_func(*args,**kwargs):
			print("call_real_func")
			func(*args,**kwargs)
		return call_real_func
	return func_replace

@dec("abc")
def realfun(arg):
	print("realfun",arg)


realfun("123")

  "@"的作用,我们通过这份代码产生的python bytecode来弄清楚@的作用

 

F:\pyweb>python -m dis decorator.py
源码行号  字节码偏移   字节码                          注释
1        0 LOAD_CONST 0 (<code object dec at 00BE74A0, file "decorator.py", line 1>)
      3 MAKE_FUNCTION 0
      6 STORE_NAME 0 (dec)
                  //到此为止是def dec(): 产生 代表dec PyFunction的字节码


12      9 LOAD_NAME 0 (dec) //加载 dec 对应的PyFunctionObject
      12 LOAD_CONST 1 ('abc')   //压入 'abc'到value-stack
      15 CALL_FUNCTION 1   //调用函数,就是dec('abc'),然后建dec 返回的 func_replace压到 value stack
                    //到此相当于 dec('abc')
      18 LOAD_CONST 2 (<code object realfun at 00BE74E8, file "decorator.py", line 12>)

                     //将代表realfun的PyCodeObject压入value stack
      21 MAKE_FUNCTION 0
                      //根据栈顶代表realfun的PyCodeObject生成 PyFunctionObject,返回之后,

                    //value stack的值为:PyFunctionObject(func_replace),PyFunctionObject(realfun)
      24 CALL_FUNCTION 1

                     //调用func_replace(realfun) ,返回call_real_func 这个PyFunctionObject压入

                    //value stack
                    //到此相当于 dec('abc')(realfun)
        27 STORE_NAME 1 (realfun) //替换掉 realfun
                    //到此相当于realfun=dec('abc')(realfun)

 

17      30 LOAD_NAME 1 (realfun)
      33 LOAD_CONST 3 ('123')
      36 CALL_FUNCTION 1
      39 POP_TOP
      40 LOAD_CONST 4 (None)
      43 RETURN_VALUE

 

 

分析之后可以知道@的作用就是在调用dec之后又添加了一个调用,而且这个调用的格式是固定的是func( realfun)这种。

所以

@dec("abc")
def realfun(arg):
  print("realfun",arg)

 

这个的实际作用是realfun=dec('abc')(realfun)。中间有一个产生PyFunctionObject,并且隐藏了调用dec返回PyFunctionObject的过程

 

另外有一种没有参数的decorator,他们省去了调用dec这一步,效果相当于realfun=dec(realfun);

 

  

 

posted @ 2016-06-14 13:13  瘸腿  阅读(351)  评论(0编辑  收藏  举报