V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
推荐学习书目
Learn Python the Hard Way
Python Sites
PyPI - Python Package Index
http://diveintopython.org/toc/index.html
Pocoo
值得关注的项目
PyPy
Celery
Jinja2
Read the Docs
gevent
pyenv
virtualenv
Stackless Python
Beautiful Soup
结巴中文分词
Green Unicorn
Sentry
Shovel
Pyflakes
pytest
Python 编程
pep8 Checker
Styles
PEP 8
Google Python Style Guide
Code Style from The Hitchhiker's Guide
fayefang
V2EX  ›  Python

Python 的装饰器执行顺序问题

  •  
  •   fayefang · 2018-09-26 19:11:22 +08:00 · 1638 次点击
    这是一个创建于 2030 天前的主题,其中的信息可能已经有所发展或是发生改变。

    我用 flask 框架做一个网站的过程中,要用到装饰器,缩减版代码如下:

    def testwra(func): print "testwra" def _wrapper(): print "wra" r = func() return r print "end" return _wrapper

    @testwra @app.route('/users', methods=['GET']) def get_users(): return ['33', '44']

    这里,正常的执行顺序不应该是 testwra->先后打印 testwra 和 end->调用_wrapper->打印 wra->调用 func()函数->返回 list->返回 r,但是我这里在浏览器中输入 url ( http://localhost:5000/users ),却不会打印 wra,好像是没有进入_wrapper 函数中。 而且还会报错 TypeError: 'list' object is not callable list 不是可以作为返回值的吗?

    9 条回复    2018-09-28 21:08:53 +08:00
    frostming
        1
    frostming  
       2018-09-27 10:26:09 +08:00
    第 0 个问题,请注意排版好吧
    第 1 个问题,app.route 会把它下面包裹的函数作为请求处理函数,这时候还没被 testwra 装饰自然没有打印 wra
    第 2 个问题,请求处理函数只能返回(字符串,response,tuple )其中一种,你返回 list 不合法,请去看 flask/app.py/make_response 源码
    frostming
        2
    frostming  
       2018-09-27 10:32:29 +08:00
    更正:请求处理函数只能返回(字符串,response,tuple , WSGI callable )其中一种
    fayefang
        3
    fayefang  
    OP
       2018-09-27 19:20:59 +08:00 via iPhone
    @frostming 装饰器不是从上向下依次执行的吗,此时 testwra 不是已经先装饰了吗,为什么说没被装饰呢
    fayefang
        4
    fayefang  
    OP
       2018-09-27 19:21:51 +08:00 via iPhone
    @frostming 我是电脑端排好版发的,发完再看排版就乱了,不知道这次什么情况
    xpresslink
        5
    xpresslink  
       2018-09-27 23:13:07 +08:00
    最重要的事情说三遍,

    装饰器是在编译时就执行,而不是调用时;装饰器只对函数进行装饰,不对装饰器进行装饰,谁贴函数近谁先。
    装饰器是在编译时就执行,而不是调用时;装饰器只对函数进行装饰,不对装饰器进行装饰,谁贴函数近谁先。
    装饰器是在编译时就执行,而不是调用时;装饰器只对函数进行装饰,不对装饰器进行装饰,谁贴函数近谁先。

    这是关键。
    sww4718168
        6
    sww4718168  
       2018-09-27 23:31:53 +08:00
    ```
    @deco
    def func():
    pass
    ```
    是 `func = deco(func)` 的语法糖。这样能明白?

    所以
    ```
    @deco1
    @deco2
    def func():
    pass
    ```
    是 `func = deco2(func); func = deco1(func)`
    也是 `func = deco1(deco2(func))`

    这样明白?
    fayefang
        7
    fayefang  
    OP
       2018-09-28 08:29:36 +08:00 via iPhone
    @sww4718168 明白了,感谢🙏🏻
    fayefang
        8
    fayefang  
    OP
       2018-09-28 08:30:02 +08:00 via iPhone
    @xpresslink 了解了,感谢🙏🏻🙏🏻
    frostming
        9
    frostming  
       2018-09-28 21:08:53 +08:00
    @fayefang 不要理解成从上到下,理解成从外到内就好了,func 在最里面
    关于   ·   帮助文档   ·   博客   ·   API   ·   FAQ   ·   我们的愿景   ·   实用小工具   ·   5817 人在线   最高记录 6543   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 27ms · UTC 06:23 · PVG 14:23 · LAX 23:23 · JFK 02:23
    Developed with CodeLauncher
    ♥ Do have faith in what you're doing.