Django 笔记

Django

Django 是 Python 下最有代表性的 Web 框架,目的是要实现简单快捷的网站开发。

对象关系映射

对象关系映射(Object Relational Mapping,简称ORM),是通过使用描述对象数据库之间映射的元数据,将面向对象语言程序中的对象自动持久化到关系数据库中。本质上就是讲数据从一种形式转换到另外一种形式。
对象和关系数据是业务实体的两种表现形式,业务实体在内存中表现为对象,在数据库中表现为关系数据。内存中的对象之间存在关联和继承关系,而在数据库中,关系数据无法直接表达多对多关联与继承关系。因此,对象-关系映射(ORM)系统一般以中间件的形式存在,主要实现程序对象到数据库数据的映射。

Django 中的对象关系映射

Django 无需数据库就可以使用,它提供了对象关系映射器。通过此技术,你可以使用 Python 代码来描述数据库结构。
models.py 中,编写类来描述你的数据模型来进行对象关系映射。

from django.db import models

class Reporter(models.Model):
    full_name = models.CharField(max_length=70)

    def __str__(self):
        return self.full_name

应用数据模型

然后,运行 Django 命令行指令来自动创建数据库表。

$ python manage.py makemigrations
$ python manage.py migrate

makemigrations指令会查看所有可用模型,并为不存在的表创建迁移。migrate执行迁移并在数据库中创造表。

享用便捷的API

接下来,你可以通过一套便捷而丰富的 Python API 访问你的数据。这些 API 是即时创建的,而不用显式的生成代码。

# 引入我们为 news app 创建的模块。
>>> from news.models import Article, Reporter

# 现在在我们系统中还不存在 Reporter。
>>> Reporter.objects.all()
<QuerySet []>

# 新建一个 Reporter。
>>> r = Reporter(full_name='John Smith')

# 通过调用 save() 方法来将对象保存到数据库中。
>>> r.save()

# 现在它有了自己的 ID。
>>> r.id
1

# 现在数据库中有一个新的 Reporter。
>>> Reporter.objects.all()
<QuerySet [<Reporter: John Smith>]>

# 可以通过访问 Python 对象的属性来访问字段。
>>> r.full_name
'John Smith'

# Django 提供了丰富的数据库查询 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.

动态管理接口

完成模型定义后,Django 会自动生成一个专业的生产机管理接口(Django admin site)——一个允许认证用户添加、更改和删除对象的 Web 站点。你只需要简单的在 admin 站点上注册你的模型即可:

from django.contrib import admin

from . import models

admin.site.register(models.Article)

创建 Django 应用的典型流程是,先建立数据模型,然后搭建管理站点,之后用户就可以向网站里填充数据。

规划 URLs

Django 推崇优美的 URL 设计。
URLconf 模块
包含一张 URL 与 Python 回调函数之间的映射表,也利于将 Python 与 URL 进行解耦。

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),
]

上述代码将 URL 路径映射到了 Python 的回调函数(“视图”)。路径字符串使用参数标签从 URL 中捕获相应值。当用户请求页面时,Django 依次遍历路径,直到初次匹配到了请求的 URL。(如果无匹配项,Django 会调用 404 视图)。这个过程非常快,因为路径在加载时就编译成了正则表达式。
一旦有 URL 路径匹配成功,Django 会调用相应的视图函数。每个视图函数会接受一个请求对象——包含请求元信息——以及在匹配式中获取的参数值。

例如,当用户请求了这样的 URL "/articles/2005/05/39323/",Django 会调用 news.views.article_detail(request, year=2005, month=5, pk=39323)

编写视图

视图函数的执行结果只可能有两种:返回一个包含请求页面元素的 HttpResponse对象,或者是抛出 Http404这类异常。至于执行过程中的其它的动作则由你决定。
通常来说,一个视图的工作就是:从参数获取数据,装载一个模板,然后将根据获取的数据对模板进行渲染。

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)

在视图里,你可以从数据库里读取记录,可以使用一个模板引擎(比如 Django 自带的,或者其他第三方的),可以生成一个 PDF 文件,可以输出一个 XML,创建一个 ZIP 文件,你可以做任何你想做的事,使用任何你想用的 Python 库。
每个视图必须要做的只有两件事:返回一个包含被请求界面内容的 HttpResponse 对象,或者抛出一个异常,比如 Http404

设计模板

Django 允许设置搜索模板路径,这样可以最小化模板之间的冗余。在 Django 设置中,你可以通过 DIRS 参数指定一个路径列表用于检索模板。如果第一个路径中不包含任何模板,就继续检查第二个,以此类推。

{% 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 %}

我们可以看到变量都被大双括号括起来了。{{ article.headline }} 的意思是“输出 article 的 headline 属性值”。
{{ article.pub_date | date:"F j, Y" }}使用了 Unix 风格的“管道符”。这是一个模板过滤器,用于过滤变量值,将对象转换为指定核实。
你可以将多个过滤器连在一起使用。还可以使用 自定义的模板过滤器,甚至可以自己编写自定义的模板标签
Django 使用了“模板继承”的概念。这就是 {{% extend "base.html %}} 的作用。它的含义是“先加载名为 base 的模板,并且用下面的标记块对模板中定义的标记块进行填充”。简而言之,模板继承可以使模板间的冗余内容最小化;每个模板只需包含与其他文档有区别的内容。

<html>
<head>
    <title>{% block title %}{% endblock %}</title>
</head>
<body>
    <img src="{% static "images/sitelogo.png" %}" alt="Logo">
    {% block content %}{% endblock %}
</body>
</html>

编写 Django 应用

安装 Django

pip install django

创建项目

cd 到你想放置代码的目录,然后运行以下命令:

$ django-admin startproject mysite

这行代码会在当前目录下创建一个 mysite 目录。

mysite/
    manage.py
    mysite/
        __init__.py
        settings.py
        urls.py
        wsgi.py

这些目录和文件的用处是:

  • 最外层的 mysite/ 根目录为项目容器。
  • manage.py 是用来管理 Django 的命令行工具。
  • 里面的一层 mysite/包含项目,它是一个纯 Python 包。她的名字就是当你引用它内部任何东西时所需要用到的 Python 包名。(eg: mysite.urls
  • mysite/__init__.py:一个空文件,告诉 Python 这个目录应该被认为是一个 Pyhton 包。
  • mysite/settings.py:Django 项目的配置文件。
  • mysite/urls.py: Django 项目的 URL 声明。
  • mysite/wsgi.py:作为你的项目的运行在 WSGI 兼容的 Web 服务器上的入口。

启动服务器

$ python manage.py runserver

这条指令启动了 Django 自带的用于开发的简易服务器,是一个用纯 Python 写的轻量级 Web 服务器。这个服务器只是为了开发而设计的,千万不要将这个服务器用于生产相关的场景中。

服务器在运行时,访问 https://127.0.0.1:8000
默认情况下,runserver 命令会将服务器设置为监听本机内部 IP 的 8000 端口。
如果你想更换服务器的监听端口,请使用命令行参数。

$ python manage.py runserver 8080

创建应用

$ python manage.py startapp [app_name]

会按照配置的项目名创建一个目录,结构如下:

[app_name]/
    __init__.py
    admin.py
    apps.py
    migrations/
        __init__.py
    models.py
    tests.py
    views.py

创建视图

在应用文件夹中view.py中进行视图编写。

URL映射

在应用文件夹下创建一个urls.py文件,用来处理应用内部的 URL 映射。

from django.urls import path

form . import views

urlpatterns = [
    path('', views.index, name='index'),
]

再将应用 URLconfs 添加到根 URLconf 中。

from django.contrib import admin
from django.urls import include, path

urlpatterns = [
    path('polls/', include('polls.urls')),
    path('admin/', admin.site.urls),
]

函数 include() 允许引用其他 URLconfs。每当 Django 遇到 include() 时,它会截断与此项匹配的 URL 的部分,并将剩余的字符串发送给具体应用的 URLconf 以供进一步处理。

当包括其它 URL 模式时你应该总是使用 include() , admin.site.urls 是唯一例外。

path()函数

此函数用于处理请求中的 url 路由分发,并调用路径所指定的视图函数。
参数:
route:route 是一个匹配 URL 的准则(类似正则表达式)。当 Django 响应一个请求时,它会从 urlpatterns 的第一项开始,按顺序依次匹配列表中的项,直到找到匹配的项。
view:当 Django 找到了一个匹配的准则,就会调用这个特定的视图函数,并传入一个 HttpRequest 对象作为第一个参数,被“捕获”的参数以关键字参数的形式传入。
kwargs:任意个关键字参数可以作为一个字典传递给目标视图函数。
name:为你的 URL 取名能使你在 Django 的任意地方唯一地引用它,尤其是在模板中。这个有用的特性允许你只改一个文件就能全局地修改某个 URL 模式。

创建模型

在 Django 里写一个数据库驱动的 Web 应用的第一步是定义模型 - 也就是数据库结构设计和附加的其它元数据。
模型是真实数据的简单明确的描述。它包含了储存的数据所必要的字段和行为。

from django.db import models


class Question(models.Model):
    question_text = models.CharField(max_length=200)
    pub_date = models.DateTimeField('date published')


class Choice(models.Model):
    question = models.ForeignKey(Question, on_delete=models.CASCADE)
    choice_text = models.CharField(max_length=200)
    votes = models.IntegerField(default=0)

每个模型被表示为 django.db.models.Model 类的子类。每个模型有一些类变量,它们都表示模型里的一个数据库字段。
每个字段都是 Field 类的实例。
每个 Field 类实例变量的名字(例如 question_textpub_date )也是字段名,所以最好使用对机器友好的格式。你将会在 Python 代码里使用它们,而数据库会将它们作为列名。
你也可以使用可选的选项为 Field 定义一个人类友好的名字。
定义 Field 类时填充的参数作为数据库字段的配置。
使用 ForeignKey 定义一个外键。

模型的用处在于:

  • 为这个应用创建数据库 schema(生成 CREATE TABLE 语句)。
  • 创建可以与对象进行交互的 Python 数据库 API。

激活应用

为了在工程中添加应用,我们需要在 settings.pyINSTALLED_APPS 添加设置。
此外,我们先来关注一下 INSTALLED_APPS 的设置项。这里包括了会在你项目中启用的所有 Django 应用。
通常, INSTALLED_APPS 默认包括了以下 Django 的自带应用:

  • django.contrib.admin -- 管理员站点。
  • django.contrib.auth -- 认证授权系统。
  • django.contrib.contenttypes -- 内容类型框架。
  • django.contrib.sessions -- 会话框架。
  • django.contrib.messages -- 消息框架。
  • django.contrib.staticfiles -- 管理静态文件的框架。

自己创建的应用或引用别人的应用,则需要在 INSTALLED_APPS 中追加。
每个应用在生成是会自动在 app.py 模块中创建包含应用名的继承于 AppConfig 类的子类。
为了在项目中包含应用,需要把项目 app.py 模块中的 AppConfig 类以点式路径的形式添加到 INSTALLED_APPS 中。

数据库配置

在项目目录下的 settings.py 中,包含了 Django 的项目设置。
Django 使用 Python 内置的 SQLite 作为默认数据库。
如果你想使用其他数据库,你需要安装合适的 database bindings,然后改变设置文件中 DATABASES'default'项目中的一些键值。

在编辑 mysite/settings.py 文件前,先设置 TIME_ZONE 为你自己时区。

数据库建表

$ python manage.py makemigrations polls

通过运行 makemigrations 命令,Django 通过检查 INSTALLED_APPS 设置,检测模型文件的修改,把修改的部分储存为一次迁移,并自动创建数据库表的修改 SQL 语句。
迁移是 Django 对于模型定义(也就是你的数据库结构)的变化的储存形式。
Django 有一个自动执行数据库迁移并同步管理你的数据库结构的命令:

$ python manage.py migrate

这个 migrate 命令选中所有还没有执行过的迁移(Django 通过在数据库中创建一个特殊的表 django_migrations 来跟踪执行过哪些迁移)并应用在数据库上 - 也就是将你对模型的更改同步到数据库结构上。

数据库操作

为了用 Python 对象展示数据表对象,Django 使用了一套直观的系统:一个模型类代表一张数据表,一个模型类的实例代表数据库表中的一行记录。

基本概念

管理器class Manager
Manager 是一种接口,它赋予了 Django 模型操作数据库的能力。
默认情况下,Django 为每个模型类添加了一个名为 objects 的 Manager。
QuerySet:一个 QuerySet 代表来自数据库中对象的一个集合。
它可以有 0 个,1 个或者多个 filters。 Filters,可以根据给定参数缩小查询结果量。
在 SQL 的层面上, QuerySet 对应 SELECT 语句,而过滤器对应类似 WHERELIMIT 的限制子句。
QuerySet 是惰性的,创建 QuerySet 并不会引发任何数据库活动。当要使用(查看,或输出)时才会从数据库中拿出。

对象创建
  • 模型对象的实例化。
    创建对象:p = Person(first_name="Bruce", last_name="Springsteen")
  • create() 方法,创建,并保存。
    p = Person.objects.create(first_name="Bruce", last_name="Springsteen")
    将修改保存至对象(存入数据库):p.save(force_insert=True)
  • get_or_create() 方法。尝试获取,不存在就创建。
对象检索

检索全部all_entries = Entry.objects.all()
返回一个包含数据库中所有对象的 QuerySet 对象。

检索单个对象one_entry = Entry.objects.get(pk=1)
get() 方法如果没有满足查询条件的结果, get()会抛出一个 DoesNotExist 异常。该异常是执行查询的模型类的一个属性。
get() 方法总是会返回一条满足条件的记录,如果不止一条记录满足其查询条件时 Django 会抛出 MultipleObjectsReturned,这同样也是模型类的一个属性。
此方法不返回 QuerySet 对象。

使用过滤器检索:都返回 QuerySet 对象。

  • filter 过滤器:filter(**kwargs)
    返回一个新的 QuerySet,包含的对象满足给定查询参数。
    ForeignKey中,你可以指定以 _id 为后缀的字段名。这种情况下,value 参数需要包含 foreign 模型的主键的原始值。
models.py
class Entry(models.Model):
    blog = models.ForeignKey(Blog, on_delete=models.CASCADE)
    --snip--

Entry.objects.filter(blog_id=4)
  • exclude 过滤器:exclude(**kwargs)
    返回一个新的 QuerySet,包含的对象 满足给定查询参数
    eg:Entry.objects.filter(pub_date__year=2006)
  • order_by()
    默认情况下,QuerySet 返回的结果都会根据模型中可以被排序的元数据所构成的元祖进行排序。你可以通过此方法来覆盖默认的排序方法。
    提供“字段名”作为参数进行升序排列,提供“-字段名”进行降序排列,提供“?”进行随机排列(但会有较大开销)
  • values()
    返回查询结果的字典结构而不是模型实例对象。提供参数则字典中只有参数字段的内容。
  • values_list()
    基本同 value(),不过返回的结果是一个元祖。

限制 QuerySet 条目数
利用 Python 的数组切片语法将 QuerySet 切成指定长度。这等价于 SQL 的 LIMITOFFSET 子句。可以再任何返回 QuerySet 对象的方法后使用。
不支持负索引。
如果检索单个对象时,使用索引,而不是切片。然而如果没有对象满足给定条件,前者会抛出 IndexError,而后者会抛出 DoesNotExist

字段查询
字段查询即你如何制定 SQL WHERE 子句。
基本的查询关键字参数遵照 field__lookuptype=value。(有个双下划线)。
例如:

Entry.objects.filter(pub_date__lte='2006-01-01')

转换为 SQL 语句大致如下:

SELECT * FROM blog_entry WHERE pub_date <= '2006-01-01';

查询子句中指定的字段必须是模型的一个字段名。

  • exact
    默认的查询类型。
    如果关键字参数未包含双下划綫——查询类型会被指定为 exact。
    因为 exaxt 类型是最为常见的。
  • iexact
    大小写不敏感的查询。
  • contains
    大小写敏感的包含匹配。相当于 SQL 中的 IS。
  • -icontains
    大小写不敏感的包含匹配。相当于 SQL 中的 LIKE。
  • in
    在给定的列表,元祖或 QuerySet 中匹配。相当于 SQL 中的 IN。
  • gt 大于
  • gte 大于等于
  • lt 小于
  • lte 小于等于
  • startswith, istartwith, endswith, iendswith
  • range(a, b) 在给定的两个范围内查询。相当于 SQL 中的 BETWEEN。
  • date 为了 date 类型的字段做匹配或比较而准备。
  • year, iso_year, month, day, week , week_day, quater, time, hour, minute, second同上
  • isnull 值取 True 或 False 时,相当于 SQL 中的 IS NULL 和 IS NOT NULL。
  • regex 大小写敏感的正则匹配
  • iregex 大小写不敏感的正则匹配

Django 管理界面

应用部署好之后,我们开始创建一个管理员页面。

$ python manage.py createsuperuser

根据指示创建好管理员页面后,启动开发服务器。

$ python manage.py runserver

打开"http://127.0.0.1:8000/admin/",你可以看到管理员登录界面。

进入站点后,可以看到 Django 管理页面的索引页。在索引页中可以进行站点的管理与模型的修改配置。

编写更多视图

在 应用的 views.py 中添加更多视图,包括接收参数的视图。
并在 urls.py 中为新增加的视图分配 url。

Django 的 URLconf 匹配机制

当你请求某一个页面时,比如说"/polls/34/",Django 会先去项目下的 settings.py 中查询 ROOT_URLCONF 中的配置。然后载入项目下的 urls 模块。然后 Django 寻找名为 urlpatterns 变量并且按序匹配正则表达式。在找到匹配项 'polls/' 后,将匹配到的文本('polls/')切除,将剩余文本 '34/' 发送到后续的 URLconf 中作进一步处理。在这里question_id=34<int:question_id> 匹配生成。使用尖括号“捕获”这部分 URL,且以关键字参数的形式发送给视图函数。上述字符串的 :question_id 部分定义了将被用于区分匹配模式的变量名,而 int: 则是一个转换器决定了应该以什么变量类型匹配这部分的 URL 路径。

模板与视图

首先在你的应用目录下创建一个 templates 目录,Django 将会在这个目录里查找模板文件。
项目中的 settings.py 模块中的 TEMPLATE 配置项描述了 Django 如何载入和渲染模板。默认的设置文件设置了 DjangoTemplates backend,并将 APP_DIRS 设置为 True。这一选项会让 DjangoTemplates 在每个 INSTALLED_APPS 文件夹中寻找 templates 子目录。

在刚创建的 template 目录中,再创建一个应用同名的文件夹,然后在其中创建模板 html 文件。

模板命名空间

虽然我们现在可以将模板文件直接放在 应用下的 templates 文件夹中(而不是再建立一个 polls 子文件夹),但是这样做不太好。Django 将会选择第一个匹配的模板文件,如果你有一个模板文件正好和另一个应用中的某个模板文件重名,Django 没有办法 区分 它们。我们需要帮助 Django 选择正确的模板,最简单的方法就是把他们放入各自的 命名空间 中,也就是把这些模板放入一个和 自身 应用重名的子文件夹里。
一段模板示例代码:

{% if latest_question_list %}
    <ul>
    {% for question in latest_question_list %}
        <li><a href="/polls/{{ question.id }}/">{{ question.question_text }}</a></li>
    {% endfor %}
    </ul>
{% else %}
    <p>No polls are available.</p>
{% endif %}

根据模板文件编写的视图代码:

from django.http import HttpResponse
from django.template import loader

from .models import Question


def index(request):
    latest_question_list = Question.objects.order_by('-pub_date')[:5]
    template = loader.get_template('polls/index.html')
    context = {
        'latest_question_list': latest_question_list,
    }
    return HttpResponse(template.render(context, request))

在上述代码中,载入 Model Question 后,从数据库中取出数据,然后载入模板文件,并且向它传递一个上下文(context)。这个上下文是一个字典,它将模板内的变量映射为 Python 对象。再将 context 和 request 通过参数传递给 template.render 方法中。

一个快捷函数:render()

「载入模板,填充上下文,再返回由它生成的 HttpResponse 对象」是一个非常常用的操作流程。于是 Django 提供了一个快捷函数—— render()
将上面的代码重构后:

from django.shortcuts import render

from .models import Question

def index(request):
    latest_question_list = Question.objects.order_by('-pub_date')[:5]
    context = {'latest_question_list': latest_question_list}
    return render(request, 'polls/index.html', context)

注意我们导入了 render 模块,取消了 loaderHttpResponse 模块的导入。

抛出 404 错误

为了网站的健壮性,我们需要对用户可能指定的非法地址抛出 Http404 异常。

from django.http import Http404
from django.shortcuts import render

from .models import Question
# ...
def detail(request, question_id):
    try:
        question = Question.objects.get(pk=question_id)
    except Question.DoesNotExist:
        raise Http404("Question does not exist")
    return render(request, 'polls/detail.html', {'question': question})

在上述代码中,如果指定的问题 ID 所对应的的问题不存在,这个视图就会抛出一个 Http 404 异常。

一个快捷函数:get_object_or_404()

尝试用 get() 函数获取一个对象,如果不存在就抛出 Http404错误也是一个普遍的流程。Django 也提供了一个快捷函数 get_object_or_404()

from django.shortcuts import get_object_or_404, render

from .models import Question
# ...
def detail(request, question_id):
    question = get_object_or_404(Question, pk=question_id)
    return render(request, 'polls/detail.html', {'question': question})

get_object_or_404() 函数获取一个 Django Model 作为它的第一个参数,然后根据之后提供的任意数量的关键字参数传递给 get() 函数并返回获取到的内容,如果这个对象于数据库中不存在,则抛出 Http404 错误。

模板常用语法

格式

Django 模板是使用 Django 模板语言标记的文本文档或 Python 字符串。模板引擎可以识别和解释这些结构,其中主要是变量和 tags。

模板是根据 view 模块中传来的 context 来转换的。转换时会在 context 中查找的变量值,替换变量,并执行 tags。其他一切按原样输出。

Django 模板语言的语法主要包含四个结构:

变量

变量 会输出一个 context 中某个键值对中的值。
变量以 {{}} 包裹。
变量的值通过句点法来获取。

标签(Tags)

Tags 提供转换(Render)过程中的任意某种逻辑。
比如,可以输出内容,使用 if操作符和for循环,从数据库中抓取内容甚至可以访问其他模板 tag。
Tag 以 {%%}包裹。

过滤器

过滤器可以对变量和 tag 参数的值进行过滤转换。
例如:
{{ django | title }}
可以把内容 {'django': 'the web framework for perfectionists with deadlines'} 转换为:
The Web Framework For Perfectionists With Deadlines

注释

{# comment #}

去除模板中的硬编码 URL

按照我们的一般逻辑,在 模板中编写链接时,链接时硬编码的:

<li><a href="/polls/{{ question.id }}/">{{ question.question_text }}</a></li>

问题在于,硬编码和强耦合的链接,对于一个包含很多应用的项目来说,修改起来是十分困难的。
然而,因为你在 polls.urlsurl() 函数中通过 name 参数为 URL 定义了名字,你可以使用 {% url %} 标签代替它:

<li><a href="{% url 'detail' question.id %}">{{ question.question_text }}</a></li>

这个标签的原理是,url函数会在 urls 模块中寻找指定了 name 的条目,在模板转换过程中用此条目的 url 进行替换。

因此,如果你想修改某个链接的 URL 时,就不需要对模板内容进行修改,而只需要对 URLconf 进行修改。

为 URL 名称添加 namespace

方法是:在根 URLconf 中添加 app_name 命名空间。

from django.contrib import admin
from django.urls import path, include

urlpatterns = [
    path('polls/', include('polls.urls', namespace='polls')),
    path('admin/', admin.site.urls),
    path('', admin.site.urls),
]

include() 函数接受一个 namespace 关键字参数,用来指定应用的 URL 命名空间。
接下来,需要在应用的 URLconf 中添加 app_name 变量。

from django.urls import path

from . import views

app_name = 'polls'
urlpatterns = [
    path('', views.index, name='index'),
    path('<int:question_id>/', views.detail, name='detail'),
    path('<int:question_id>/results/', views.results, name='results'),
    path('<int:question_id>/vote/', views.vote, name='vote'),
]

接下来对模板进行修改,将链接指向具有命名空间的视图。

<li><a href="{% url 'polls:detail' question.id %}">{{ question.question_text }}</a></li>

表单

一个表单模板示例:

<h1>{{ question.question_text }}</h1>

{% if error_message %}<p><strong>{{ error_message }}</strong></p>{% endif %}

<form action="{% url 'polls:vote' question.id %}" method="post">
{% csrf_token %}
{% for choice in question.choice_set.all %}
    <input type="radio" name="choice" id="choice{{ forloop.counter }}" value="{{ choice.id }}">
    <label for="choice{{ forloop.counter }}">{{ choice.choice_text }}</label><br>
{% endfor %}
<input type="submit" value="Vote">
</form>

模板中设置了表单的 action urlmethod = "post"
forloop.counter 表示 'for' 循环的循环计数。Django 提供了一系列便于监控 for 循环的变量。

  • forloop.counter 当前循环的迭代次数(从 1 开始计数)
  • forloop.counter0 当前循环的迭代次数(从 0 开始计数)
  • forloop.revcounter 当前循环距离循环结束的计数(从 1 开始)
  • forloop.revcounter0 当前循环距离循环结束的计数(从 0 开始)
  • forloop.first 如果是第一次执行循环,返回 True。
  • forloop.last 如果是最后一次执行循环,返回 False。
  • forloop.parentloop 当存在嵌套循环时,指向当前循环的外层循环。

Django 为 POST 表单提供了 {% csrf_token %} 模板标签,防止跨站攻击。因此针对内部 URL 的 POST 表单都应该使用这个标签。

一个表单处理示例:

from django.http import HttpResponse, HttpResponseRedirect
from django.shortcuts import get_object_or_404, render
from django.urls import reverse

from .models import Choice, Question
# ...
def vote(request, question_id):
    question = get_object_or_404(Question, pk=question_id)
    try:
        selected_choice = question.choice_set.get(pk=request.POST['choice'])
    except (KeyError, Choice.DoesNotExist):
        # Redisplay the question voting form.
        return render(request, 'polls/detail.html', {
            'question': question,
            'error_message': "You didn't select a choice.",
        })
    else:
        selected_choice.votes += 1
        selected_choice.save()
        # Always return an HttpResponseRedirect after successfully dealing
        # with POST data. This prevents data from being posted twice if a
        # user hits the Back button.
        return HttpResponseRedirect(reverse('polls:results', args=(question.id,)))

request.POST 是一个类字典对象,让你可以通过关键字的名字获取提交的数据。这个例子中, request.POST['choice'] 以字符串形式返回选择的 Choice 的 ID。
request.POST 的值永远是字符串。
request.GETrequest.POST 的情况类似。

如果在 request.POST['choice'] 数据中没有提供 choice,POST 将引发一个 KeyError

在请求执行完毕后,代码返回一个 HttpResponseRedirect 而不是常用的 HttpResponseHttpResponseRedirectHttpResponse 的一个子类,只接收一个参数:用户将要被重定向的 URL。

在这个例子中,我们在 HttpResponseRedirect 的构造函数中使用 reverse() 函数。这个函数避免了我们在视图函数中硬编码 URL。
reverse() 函数用来拼接 url 字符串,第一个参数需要提供 viewname,它可以是一个 URL 模板的 name。如果 URL 接受参数,你可以通过 args 作为第二个参数来传递,同样你可以使用 kwargs,但你不能两者同时使用。

减少模板代码冗余——通用视图系统

基本的 Web 开发中有一个常见情况:根据 URL 中的参数从数据库中获取数据、载入模板文件然后返回渲染后的模板。 由于这种情况特别常见,Django 提供一种快捷方式,叫做“通用视图”系统。
通用视图将常见的模式抽象化,可以使你在编写应用时甚至不需要编写 Python 代码。
URL 配置:

path('', views.IndexView.as_view(), name='index'),
path('<int:pk>/', views.DetailView.as_view(), name='detail'),

视图:

class IndexView(generic.ListView):
    template_name = 'polls/index.html'
    context_object_name = 'latest_question_list'

    def get_queryset(self):
        """Return the last five published questions."""
        return Question.objects.order_by('-pub_date')[:5]


class DetailView(generic.DetailView):
    model = Question
    template_name = 'polls/detail.html'

在上述代码中,使用了两个通用视图:ListViewDetailView。这两个视图分别抽象“显示一个对象列表”和“显示一个特定类型对象的详细信息页面”这两种概念。

  • 每个通用视图需要知道它将作用于哪个模型。这由 model 属性提供。
  • DetailView 期望从 URL 中不活名为 "pk" 的主键值。

template_name 模板是用来告诉 Django 使用一个指定的模板名字,而不是默认模板名字。默认情况下,通用视图 DetailView 使用一个叫做 <appname>/<module name>_detail.html 的模板。在我们的例子中,它默认会使用 "polls/question_detail.html" 模板。
类似的,ListView 会使用一个叫做 <app name>/<model name>_list.html 的默认模板;我们使用 template_name 来告诉 ListView 使用我们创建的已经存在的 "polls/index.html" 模板。

之前,提供模板文件时会带有一个包含 questionlatest_question_list 变量的 context
对于 DetailView , question 变量会自动提供—— 因为我们提供 model = Question ,为 Django 提供了我们的模型 (Question), Django 能够为 context 变量决定一个合适的名字。
然而对于 ListView, 自动生成的 context 变量是 question_list。为了覆盖这个行为,我们提供 context_object_name 属性,表示我们想使用 latest_question_list。作为一种替换方案,你可以改变你的模板来匹配新的 context 变量 —— 这是一种更便捷的方法,告诉 Django 使用你想使用的变量名。

自动化测试

测试,是用来检查代码正确性的一些简单的程序。
自动化测试是有某个系统帮你自动完成的。当你创建好了一系列测试,每次修改应用代码后,就可以自动检查出修改后的代码是否还像你曾经预期的那样正常工作。你不需要花费大量时间来进行手动测试。

创建一个 tests 模块。

import datetime

from django.test import TestCase
from django.utils import timezone

from .models import Question


class QuestionModelTests(TestCase):

    def test_was_published_recently_with_future_question(self):
        """
        was_published_recently() returns False for questions whose pub_date
        is in the future.
        """
        time = timezone.now() + datetime.timedelta(days=30)
        future_question = Question(pub_date=time)
        self.assertIs(future_question.was_published_recently(), False)

运行测试

$ python manage.py test polls
  • python manage.py test polls 将会寻找 polls 应用里的测试代码
  • 它找到了 django.test.TestCase 的一个子类
  • 它创建一个特殊的数据库供测试使用
  • 它在类中寻找测试方法——以 test 开头的方法。
  • test_was_published_recently_with_future_question 方法中,它创建了一个 pub_date 值为 30 天后的 Question 实例。
  • 接着使用 assertls() 方法,发现 was_published_recently() 返回了 True,而我们期望它返回 False
  • python manage.py test polls 将会寻找 polls 应用里的测试代码
  • 它找到了 django.test.TestCase 的一个子类
  • 它创建一个特殊的数据库供测试使用
  • 它在类中寻找测试方法——以 test 开头的方法。
  • test_was_published_recently_with_future_question 方法中,它创建了一个 pub_date 值为 30 天后的 Question 实例。
  • 接着使用 assertls() 方法,发现 was_published_recently() 返回了 True,而我们期望它返回 False

针对视图的测试

测试工具: Client

静态文件管理

django.contrib.staticfiles:用来将各个应用的静态文件统一收集起来并分发。

自定义应用的界面和风格

在应用目录下创建一个名为 static 的目录。Django 将在目录下查找静态文件。

Django 的 STATICFILES_FINDERS 设置包含了一系列的查找器,他们知道去哪里找到 static 文件。
AppDirectoriesFinder 是默认查找器中的一个,它会在每个 INSTALLED_APPS 中指定的应用的子文件中寻找名称为 static 的特定文件夹。
管理后台采用相同的目录结构管理它的静态文件。
所以你应该将所有的静态文件以 static/应用名/文件 的路径存储。因为 AppDirectoriesFinder 的存在,你可以在 Django 中简单的使用 应用名/文件 的形式引用此文件,类似于你引用模板路径的方式。

在相应的目录下创建文件后,可以在需要引入的模块中以如下方式引用该文件。

{% load static %}

<link rel="stylesheet" type="text/css" href="{% static 'polls/style.css' %}">

{% static %} 模板标签会生成静态文件的绝对路径。

添加背景图

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,794评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,050评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,587评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,861评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,901评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,898评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,832评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,617评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,077评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,349评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,483评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,199评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,824评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,442评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,632评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,474评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,393评论 2 352

推荐阅读更多精彩内容