Django at a glance
Because Django was developed in a fast-paced newsroom environment, it wasdesigned to make common Web-development tasks fast and easy. Here’s an informaloverview of how to write a database-driven Web app with Django.
The goal of this document is to give you enough technical specifics tounderstand how Django works, but this isn’t intended to be a tutorial orreference – but we’ve got both! When you’re ready to start a project, you canstart with the tutorial or dive right into moredetailed documentation.
Design your model
Although you can use Django without a database, it comes with anobject-relational mapper in which you describe your database layout in Pythoncode.
The data-model syntax offers many rich ways ofrepresenting your models – so far, it’s been solving many years’ worth ofdatabase-schema problems. Here’s a quick example:
- from django.db import models
- class Reporter(models.Model):
- full_name = models.CharField(max_length=70)
- def __str__(self):
- return self.full_name
- class Article(models.Model):
- pub_date = models.DateField()
- headline = models.CharField(max_length=200)
- content = models.TextField()
- reporter = models.ForeignKey(Reporter, on_delete=models.CASCADE)
- def __str__(self):
- return self.headline
Install it
Next, run the Django command-line utilities to create the database tablesautomatically:
- $ python manage.py makemigrations
- $ python manage.py migrate
- ...\> py manage.py makemigrations
- ...\> py manage.py migrate
The makemigrations
command looks at all your available models andcreates migrations for whichever tables don’t already exist. migrate
runs the migrations and creates tables in your database, as well as optionallyproviding much richer schema control.
Enjoy the free API
With that, you’ve got a free, and rich, Python APIto access your data. The API is created on the fly, no code generationnecessary:
- # Import the models we created from our "news" app
- >>> from news.models import Article, Reporter
- # No reporters are in the system yet.
- >>> Reporter.objects.all()
- <QuerySet []>
- # Create a new Reporter.
- >>> r = Reporter(full_name='John Smith')
- # Save the object into the database. You have to call save() explicitly.
- >>> r.save()
- # Now it has an ID.
- >>> r.id
- 1
- # Now the new reporter is in the database.
- >>> Reporter.objects.all()
- <QuerySet [<Reporter: John Smith>]>
- # Fields are represented as attributes on the Python object.
- >>> r.full_name
- 'John Smith'
- # Django provides a rich database lookup API.
- >>> Reporter.objects.get(id=1)
- <Reporter: John Smith>
- >>> Reporter.objects.get(full_name__startswith='John')
- <Reporter: John Smith>
- >>> Reporter.objects.get(full_name__contains='mith')
- <Reporter: John Smith>
- >>> Reporter.objects.get(id=2)
- Traceback (most recent call last):
- ...
- DoesNotExist: Reporter matching query does not exist.
- # Create an article.
- >>> from datetime import date
- >>> a = Article(pub_date=date.today(), headline='Django is cool',
- ... content='Yeah.', reporter=r)
- >>> a.save()
- # Now the article is in the database.
- >>> Article.objects.all()
- <QuerySet [<Article: Django is cool>]>
- # Article objects get API access to related Reporter objects.
- >>> r = a.reporter
- >>> r.full_name
- 'John Smith'
- # And vice versa: Reporter objects get API access to Article objects.
- >>> r.article_set.all()
- <QuerySet [<Article: Django is cool>]>
- # The API follows relationships as far as you need, performing efficient
- # JOINs for you behind the scenes.
- # This finds all articles by a reporter whose name starts with "John".
- >>> Article.objects.filter(reporter__full_name__startswith='John')
- <QuerySet [<Article: Django is cool>]>
- # Change an object by altering its attributes and calling save().
- >>> r.full_name = 'Billy Goat'
- >>> r.save()
- # Delete an object with delete().
- >>> r.delete()
A dynamic admin interface: it’s not just scaffolding – it’s the whole house
Once your models are defined, Django can automatically create a professional,production ready administrative interface –a website that lets authenticated users add, change and delete objects. Theonly step required is to register your model in the admin site:
- from django.db import models
- class Article(models.Model):
- pub_date = models.DateField()
- headline = models.CharField(max_length=200)
- content = models.TextField()
- reporter = models.ForeignKey(Reporter, on_delete=models.CASCADE)
- from django.contrib import admin
- from . import models
- admin.site.register(models.Article)
The philosophy here is that your site is edited by a staff, or a client, ormaybe just you – and you don’t want to have to deal with creating backendinterfaces only to manage content.
One typical workflow in creating Django apps is to create models and get theadmin sites up and running as fast as possible, so your staff (or clients) canstart populating data. Then, develop the way data is presented to the public.
Design your URLs
A clean, elegant URL scheme is an important detail in a high-quality Webapplication. Django encourages beautiful URL design and doesn’t put any cruftin URLs, like .php
or .asp
.
To design URLs for an app, you create a Python module called a URLconf. A table of contents for your app, it contains a mappingbetween URL patterns and Python callback functions. URLconfs also serve todecouple URLs from Python code.
Here’s what a URLconf might look like for the Reporter
/Article
example above:
- from django.urls import path
- from . import views
- urlpatterns = [
- path('articles/<int:year>/', views.year_archive),
- path('articles/<int:year>/<int:month>/', views.month_archive),
- path('articles/<int:year>/<int:month>/<int:pk>/', views.article_detail),
- ]
The code above maps URL paths to Python callback functions (“views”). The pathstrings use parameter tags to “capture” values from the URLs. When a userrequests a page, Django runs through each path, in order, and stops at thefirst one that matches the requested URL. (If none of them matches, Djangocalls a special-case 404 view.) This is blazingly fast, because the paths arecompiled into regular expressions at load time.
Once one of the URL patterns matches, Django calls the given view, which is aPython function. Each view gets passed a request object – which containsrequest metadata – and the values captured in the pattern.
For example, if a user requested the URL “/articles/2005/05/39323/”, Djangowould call the function news.views.article_detail(request,year=2005, month=5, pk=39323)
.
Write your views
Each view is responsible for doing one of two things: Returning anHttpResponse
object containing the content for therequested page, or raising an exception such as Http404
.The rest is up to you.
Generally, a view retrieves data according to the parameters, loads a templateand renders the template with the retrieved data. Here’s an example view foryear_archive
from above:
- from django.shortcuts import render
- from .models import Article
- def year_archive(request, year):
- a_list = Article.objects.filter(pub_date__year=year)
- context = {'year': year, 'article_list': a_list}
- return render(request, 'news/year_archive.html', context)
This example uses Django’s template system, which hasseveral powerful features but strives to stay simple enough for non-programmersto use.
Design your templates
The code above loads the news/year_archive.html
template.
Django has a template search path, which allows you to minimize redundancy amongtemplates. In your Django settings, you specify a list of directories to checkfor templates with DIRS
. If a template doesn’t existin the first directory, it checks the second, and so on.
Let’s say the news/year_archive.html
template was found. Here’s what thatmight look like:
mysite/news/templates/news/year_archive.html
- {% extends "base.html" %}
- {% block title %}Articles for {{ year }}{% endblock %}
- {% block content %}
- <h1>Articles for {{ year }}</h1>
- {% for article in article_list %}
- <p>{{ article.headline }}</p>
- <p>By {{ article.reporter.full_name }}</p>
- <p>Published {{ article.pub_date|date:"F j, Y" }}</p>
- {% endfor %}
- {% endblock %}
Variables are surrounded by double-curly braces. {{ article.headline }}
means “Output the value of the article’s headline attribute.” But dots aren’tused only for attribute lookup. They also can do dictionary-key lookup, indexlookup and function calls.
Note {{ article.pub_date|date:"F j, Y" }}
uses a Unix-style “pipe” (the “|”character). This is called a template filter, and it’s a way to filter the valueof a variable. In this case, the date filter formats a Python datetime object inthe given format (as found in PHP’s date function).
You can chain together as many filters as you’d like. You can write customtemplate filters. You can writecustom template tags, which run customPython code behind the scenes.
Finally, Django uses the concept of “template inheritance”. That’s what the{% extends "base.html" %}
does. It means “First load the template called‘base’, which has defined a bunch of blocks, and fill the blocks with thefollowing blocks.” In short, that lets you dramatically cut down on redundancyin templates: each template has to define only what’s unique to that template.
Here’s what the “base.html” template, including the use of static files, might look like:
- {% load static %}
- <html>
- <head>
- <title>{% block title %}{% endblock %}</title>
- </head>
- <body>
- <img src="{% static "images/sitelogo.png" %}" alt="Logo">
- {% block content %}{% endblock %}
- </body>
- </html>
Simplistically, it defines the look-and-feel of the site (with the site’s logo),and provides “holes” for child templates to fill. This means that a site redesigncan be done by changing a single file – the base template.
It also lets you create multiple versions of a site, with different basetemplates, while reusing child templates. Django’s creators have used thistechnique to create strikingly different mobile versions of sites by onlycreating a new base template.
Note that you don’t have to use Django’s template system if you prefer anothersystem. While Django’s template system is particularly well-integrated withDjango’s model layer, nothing forces you to use it. For that matter, you don’thave to use Django’s database API, either. You can use another databaseabstraction layer, you can read XML files, you can read files off disk, oranything you want. Each piece of Django – models, views, templates – isdecoupled from the next.
This is just the surface
This has been only a quick overview of Django’s functionality. Some more usefulfeatures:
- A caching framework that integrates with memcachedor other backends.
- A syndication framework that lets youcreate RSS and Atom feeds by writing a small Python class.
- More attractive automatically-generated admin features – this overviewbarely scratched the surface.The next steps are for you to download Django, read the tutorial and join the community. Thanks for your interest!