Lesson 4 使用模板显示内容

模型的Objects获取或操作模型的对象

1. views.py要怎么写

第三课创建好应用后,打开应用目录显示如下:

目录结构

其中,templates文件夹和urls.py是我后创建的。其他都是一开始就有的。

在经过上节课一番折腾后,我们来到了写views.py的环节。这次views.py是已经自带了,不需要手动创建。初始代码如下:

# views.py
from django.shortcuts import render

# Create your views here

除了import了一个模块之后便再无其他内容。需要我们手写。

先去models.py定义model如下:

#!/usr/bin/env python3
# -*- coding: utf-8 -*-

from django.db import models

# Create your models here.
class Article(models.Model): # 从models中找字段类型
    title = models.CharField(max_length = 30)
    content = models.TextField()

然后回到views.py,重新定义:

from django.shortcuts import render
from .models import Article
from django.http import HttpResponse

# Create your views here
def article_detail(request,  article_id):
    article = Article.objects.get(id = article_id)
    return HttpResponse("文章内容是{} <br> 标题是{}".format(article.title, article.content))

这样一来,我们就可以在页面中显示文章内容与标题了。但由于Django是后端框架,这样写反而是前后端代码混一起了,违背了我们“前后端分离”的原则,所以怎样把前端代码从后端中剥离出来呢?

首先,新建templates文件夹,用于存放html代码。这是Django固定语法,不要叫别的名字

然后,根据实际情况建立对应的html文件。比如,显示文章细节,则需要article_detail.html。这时候默认引入的render方法开始起作用了。我们要将HttpResponse("文章内容是{} <br> 标题是{}".format(article.title, article.content))这一行代码换成render方法可以做如下变换:

from django.shortcuts import render
from .models import Article

# Create your views here
def article_detail(request,  article_id):
    article = Article.objects.get(id = article_id)
    context = dict()
    context["article"] = article
    return render(request, "article_detail.html", context)

同时,在templates文件夹下的article_detail.html可以这样写:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Article Detail</title>
</head>
<body>
    <h2>{{ article.title }}</h2> 
    <!---article.title就是原models.py中的article对象属性,用两个圆括号括起来--->
    <hr>
    <p>{{ article.content }}</p>
</body>
</html>

可以看到上面的html代码中的<h2>{{ article.title }}</h2>一行。这一行是Django模板的应用。更多关于模板的可以点这里。下面一行还有模板。

但这样写的views.py还有一个问题。就是当文章不存在时,怎么返回404信息?于是我们发现django.http有一个Http404方法可以使用,于是我们可以引入它。

from django.http import Http404

经过优化后,代码可以变成如下形式:

# views.py
from django.shortcuts import render
from django.http import Http404
from .models imprt Article

# Create your views here
def article_detail(request,  article_id):
    try:
        article = Article.objects.get(id = article_id)
        context = dict()
        context["article"] = article
        return render(request, "article_detail.html", context)
    except Article.DoesNotExist:
        raise Http404("文章不存在!")

但这样写的话依然很复杂,有没有一个方法,可以在正常时返回结果,错误时返回404呢?有的。get_object_or_404可以达到这一点。

from django.shortcuts import get_object_or_404

同样,render方法可以被简化为:

from django.shortcuts import render_to_response

通过一通操作后,我们的代码可以简化为:

# views.py
from django.shortcuts import render_to_response, get_object_or_404
from .models imprt Article

# Create your views here
def article_detail(request,  article_id):
    article = get_object_or_404(Article, pk = article_id)
    context = dict()
    context["article"] = article
    return render_to_response("article_detail.html", context)

如果我们还想定义一个文章列表的view,那么可以这样操作:

# views.py
from django.shortcuts import render_to_response, get_object_or_404
from .models imprt Article

# Create your views here
def article_detail(request,  article_id):
    article = get_object_or_404(Article, pk = article_id)
    context = dict()
    context["article"] = article
    return render_to_response("article_detail.html", context)

def article_list(request):
    articles = Article.objects.all() # 用对象.objects方法
    context = dict()
    context["articles"] = articles
    return render_to_response("article_list.html", context)

同理,我们还需要定义article_list.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Article List</title>
</head>
<body>
    {% for article in articles %}
        <a href = {% url "article_id" article.pk %}>{{ article.title }}</a>
    {% endfor %}
</body>
</html>

中间的这段代码非常显眼:

    {% for article in articles %}
        <a href = {% url "article_id" article.pk %}>{{ article.title }}</a>
    {% endfor %}
  • {% %}这是模板语法。
  • {% for %}:模板中使用for循环的标志,常与{% endfor %}搭配使用。
  • {% url %}:url标签。后面两个参数,article_id是我们在urls.py中指定的别名。path("article/<int:article_id>", article_detail, name = "article_id"),。这就是前面为什么urls.py中为什么要指定name参数的原因了。article.pk指的是article = get_object_or_404(Article, pk = article_id)中的pk = article_id而写的。这个标签意思是article.title显示标题,链接指向具体的文章。

于是乎,这样一来我们就实现了前后端分离。

2. 分会场路由

目录结构

我在article中另创建了一个urls.py,意思是每个应用都应有每个应用的路由。这样一来就不用总是去总应用中的urls.py中改来改去了。

这样做的好处是每个应用都可以创建自己的路由,不用改来改去了,节省代码量。

先看一下总应用中怎样包括分应用路由的说明:

Including another URLconf

  1. Import the include() function: from django.urls import include, path
  2. Add a URL to urlpatterns: path('blog/', include('blog.urls'))

翻译一下:

  • 开头引入include()函数;
  • urlpatterns列表中可以写元素如右:path('blog/', include('blog.urls'))

于是乎,我们的总会场的urls.py可以修改如下:

from django.contrib import admin
# from django.urls import path
from django.urls import include,path # 引入include函数
from .views import index
# from article.views import article_detail, article_list

urlpatterns = [
    path('admin/', admin.site.urls),
    path("", index),
    # path("article/<int:article_id>", article_detail, name = "article_id"), #原代码,被我注释掉了
    # path("article/", article_list, name = "article_list"), # 原代码,被我注释掉了
    path("article/", include("article.urls")), # 新代码,可以看到我们用了include函数之后是什么样的
]

那么,分会场的urls.py要怎么写呢?

  • 首先,仿照总会场,from django.urls import path和定义urlpatterns列表是必须的。
  • 然后,引入本应用的views。可以写作from . import views
  • 最后,改一下总会场被我注释的两行代码即可写成。

最终效果如下:

#!/usr/bin/env python3
# -*- coding: utf-8 -*-

from django.urls import path
from . import views

urlpatterns = [
    path("<int:article_id>", views.article_detail, name="article_id"), # 相对路径即可
    path("", views.article_list, name="article_list"), # 已经在总会场指定了分会场的上级路由是"articles/"了,就不需要做重复工作了
]

这样一来,分会场的urls.py便定义完成了。

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