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(
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.
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/
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.