how do decorated functions work in flask/python? (app.route)

后端 未结 2 1714
一向
一向 2020-12-30 07:56

Edit 2: I misunderstood how decorators work. The decorator is run even if the decorated function is not called (even though you might not see it\'s effects). function = dec(

相关标签:
2条回答
  • 2020-12-30 08:35

    Your code snippet is calling the hello function directly. That's why the function is running.

    For Flask, you define the routes using decorator functions. When you navigate to a URL that matches the decorator, it will execute the decorated function. In this example, if I navigate to "/" on my webserver, the hello function will execute.

    To actually run your server, you need to do the following from the command line (which can also be found in Flask's documentation.

    $ export FLASK_APP=hello.py
    $ flask run
    

    Where hello.py is the name of the file containing your Python code. You should also remove the direct call to hello() in your file. Then open a browser and navigate to http://localhost:5000/ to see the result.

    How do Decorators Work

    Decorators are functions that wrap other functions in an attempt to alter the behavior of the sub-function. You can read the very detailed explanation in the Python wiki.

    Decorators take in a function as their argument. Typically, decorators run some code before executing the sub-function. For example, if you want to add authentication to certain endpoints of your Flask app, you can use decorators. The decorators check to make sure a user is authenticated to use a resource before the actual code of that resource executes.

    @authenticate
    def hello():
       return "Hello world"
    

    The authenticate function will run first. And then if the user is authenticated, it will call hello to execute the rest, otherwise, it will send an error back to the user. In the case of Flask's decorators, they are checking to see if an incoming request matches the route you've specified. If it does, then it executes the function. Otherwise, it checks the next route.

    The following article is my go to for learning about decorators: https://realpython.com/blog/python/primer-on-python-decorators/

    0 讨论(0)
  • 2020-12-30 08:39

    Python decorators like this:

    @decorator
    def func():
        pass
    

    Can be changed to look like this instead:

    def func():
        pass
    
    decorator(func)
    

    Or in other words, they're functions that take functions. In some circumstances, you might not see the effects of the decorator immediately, so it may seem like the decorator itself isn't used until the function it decorates is called, but that's not an actual restriction of Python decorators.

    In your code, @app.route("/") is a decorator which adds an endpoint to the app object. It doesn't actually modify any behavior of the function, and is instead sugar to simplify the process. Without the route() decorator, this is how you'd do the equivalent route registration in Flask.

    from flask import Flask
    app = Flask(_name_)
    
    def hello():
        return "Hello world"
    
    app.add_url_rule("/", "hello", hello)
    

    And if you look at the implementation of the route decorator in Flask, you'll see that this is the equivalent.

    def route(self, rule, **options):
            """A decorator that is used to register a view function for a
            given URL rule.  This does the same thing as :meth:`add_url_rule`
            but is intended for decorator usage::
                @app.route('/')
                def index():
                    return 'Hello World'
            For more information refer to :ref:`url-route-registrations`.
            :param rule: the URL rule as string
            :param endpoint: the endpoint for the registered URL rule.  Flask
                             itself assumes the name of the view function as
                             endpoint
            :param options: the options to be forwarded to the underlying
                            :class:`~werkzeug.routing.Rule` object.  A change
                            to Werkzeug is handling of method options.  methods
                            is a list of methods this rule should be limited
                            to (``GET``, ``POST`` etc.).  By default a rule
                            just listens for ``GET`` (and implicitly ``HEAD``).
                            Starting with Flask 0.6, ``OPTIONS`` is implicitly
                            added and handled by the standard request handling.
            """
            def decorator(f):
                endpoint = options.pop('endpoint', None)
                self.add_url_rule(rule, endpoint, f, **options)
                return f
            return decorator
    

    So you can see that route adds it to the app's router, so once the Flask app gets a request, it decides how to execute code/view for the endpoint that's been requested.

    0 讨论(0)
提交回复
热议问题