Django学习(3)

创建第一个 Django 应用程序时,根据第 3 部分的内容进行尝试和调查,并记录下来。前一篇文章是关于学习 Django(2)的内容。

不过,这种愚蠢的事情我们还是不要做了。在Django学习(1)中尝试并跳过了这部分。

相反的是,我意识到在这之后,如果数据库中没有数据,就会变得无趣,所以我将数据输入。

管理员

要将数据写入数据库,可以通过Db Browser for SQlite直接将数据写入数据库,或者编写一个将数据写入数据库的视图(通常是使用这种方法)。然而,还有一种方法是成为超级用户,通过管理员界面来设置数据。

创建超级用户

使用以下命令创建超级用户

用python manage.py createsuperuser命令创建超级用户。

Username (leave blank to use 'akira'): admin
Email address: test@hogehoge.jp
Password: ********
Password (again): ********
Superuser created successfully.

如果密码太过简单易猜,将可能被盗取。

接下来,在管理界面中编辑 polls 应用的模型,将 polls/admin.py 文件修改为以下内容。

from django.contrib import admin

from .models import Question, Choice

admin.site.register(Question)
admin.site.register(Choice)

自从进行了这项更改之后

执行 `python manage.py runserver`

Django003.jpg

在POLLS下面可以看到两个选项:Choises和Questions。这些选项与在models.py中创建的类相对应。

点击“追加”按钮,可以添加数据。
由于外观和models.py中设置的字段完全相同,所以即使没有疑问也可以添加数据。

更改视图并查看数据

暂时先用教程中的代码试试看。虽然说了这话,但因为之前已经加了很多内容,所以可能会带有多余的代码。

from django.http import HttpResponse
from django.urls import reverse

from .models import Question

def index(request):
    latest_question_list = Question.objects.order_by('-pub_date')[:5]
    output = ', '.join([q.question_text for q in latest_question_list])
    return HttpResponse(output)

    # Leave the rest of the views (detail, results, vote) unchanged

def subView(request, number):
    urlName = reverse('suburl', args=[number])
    return HttpResponse("これはsubViewです。{0} url={1}".format(number, urlName))

def repathView(request, number):
    urlName = reverse('re_path_url', args=[number])
    return HttpResponse("repathView。{0} url={1}".format(number, urlName))

当您访问http://127.0.0.1:8000/polls/并进行此更改后,将显示Question的question_text。

我稍微研究一下这个处理内容。

latest_question_list = Question.objects.order_by('-pub_date')[:5]

所以,可以想象正在加载数据库,但为什么是Question.objects.order_by呢,这是为了获取对象而如下所示的检索。

通过使用模型的Manager来获取QuerySet。每个模型至少有一个Manager,并且默认情况下命名为objects。请直接从模型类访问如下所示。

关于QuerySet的内容已在QuerySet API中有所记录,并且对order_by的解释也同时在此处记录了。

根据Question.objects.order_by调用,从数据库中读取Question表的数据。

output = ', '.join([q.question_text for q in latest_question_list])

所以,创建一个以逗号分隔的字符串,其中包含了Question表的question_text字段

return HttpResponse(output)

所以,输出这个字符串。
当使用HttpResponse()返回字符串时,它会被发送到浏览器端。

既然这么辛苦,何不试着表达出来呢?

试图将 def subView(request, number) 改造成一个表格(其他部分保持不变)。

from pytz import timezone

def subView(request, number):
    outStr1 ="<html>"\
            "<head>"\
            "<title>Question DBの中味を表示する</title>"\
            "</head>"\
            "<body>"\
            "<table>"\

    outStr2 ="</table>"\
             "</body>"\
             "</html>"

    latest_question_list = Question.objects.order_by('-pub_date')
    outList = ""

    for q in latest_question_list:
        print(q.pub_date.astimezone(timezone('Asia/Tokyo')))
        outList += "<tr><td>" + q.pub_date.astimezone(timezone('Asia/Tokyo')).strftime( '%Y/%m/%d %H:%M:%S') +"</td><td>" + q.question_text + "</td></tr>"
    output = outStr1+ outList + outStr2
    return HttpResponse(output)

使用模板

由于在硬编码字符串中编写HTML不美观,所以应该使用模板。

在<项目名称>/settings.py文件中有以下设置

INSTALLED_APPS = [
    'polls.apps.PollsConfig',
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
]
      
      
      

TEMPLATES = [
    {
        'BACKEND': 'django.template.backends.django.DjangoTemplates',
        'DIRS': [],
        'APP_DIRS': True,
        'OPTIONS': {
            'context_processors': [
                'django.template.context_processors.debug',
                'django.template.context_processors.request',
                'django.contrib.auth.context_processors.auth',
                'django.contrib.messages.context_processors.messages',
            ],
        },
    },
]

根据这个设置中的 ‘APP_DIRS’: True ,Django可以在INSTALLED_APPS中注册的应用程序目录中搜索模板。这是在教程中提到的。

所以,在polls目录下创建一个名为templates/pollsTemp的目录,并放置模板文件。
为什么不放在templates目录下呢?在教程中提到了模板的命名空间,也就是说Django在获取模板文件时,并不区分templates目录所属的应用程序,因此当应用程序之间存在名称冲突时,就会导致意外选择了模板文件。
相反,我认为在templates下创建的目录名称并不需要与应用程序名称相同(尽管相同的更易于理解),所以我故意改变了名称来尝试一下。

尽管与教程不同,但为了生成与刚刚编写的HTML完全相同的输出,现在尝试创建一个模板文件。
因此,创建一个名为polls/templates/pollsTemp/repathView.html的文件。

模板的语法在Django模板语言中有详细介绍。

变量
变量在模板中以 {{ 变量 }} 的形式进行描述。
虽然文档上写了很多,但是目前只理解这一点。

在中国提供了一种选项来用中国本地语言重述上述内容:
可以指定变量显示的格式
使用{{变量 | 过滤器}}的形式进行描述。
有关可用的内置过滤器,请参阅内置过滤器参考文档。

前述的strftime格式对应的日期显示为{{ 变量 | date:”Y/m/d ” }}{{ 变量 | time:”H:i:s” }}。

标签
标签用来表示控制语法等。在标签中不能写入Python语法,而是使用此模板专用的语法,并以{% 标签 %}的格式进行编写。
有关预设的内建标签的信息可以在内建标签参考中找到。

for循环的重复如下所示

{% for 变量 in 列表变量 %}

{% endfor %}

如果条件一成立,则执行表达式一;如果条件一不成立但条件二成立,则执行表达式二;如果条件一和条件二都不成立,则执行表达式三的分支。

如果语句是 “如果条件一成立,则执行表达式一;如果条件一不成立但条件二成立,则执行表达式二;如果条件一和条件二都不成立,则执行表达式三的分支” ,这可以用以下方式表达:

如果条件一{% if 条件式1 %} {% endif %} 成立,则执行表达式一,如果条件一不成立但条件二{% elif 条件式2 %} {% endif %} 成立,则执行表达式二,否则{% else %} {% endif %} 执行表达式三的分支。

根据此进行模板的创建。

<!DOCTYPE html>
<html lang="ja">
<head>
    <meta charset="UTF-8">
    <title>Question DBの中味を表示する</title>
</head>
<body>

{% if latest_question_list %}
<table>
    {% for q in latest_question_list %}
    <tr>
        <td>{{q.pub_date | date:"Y/m/d " }}{{q.pub_date | time:"H:i:s" }}</td>
        <td>{{ q.question_text }}</td>
    </tr>
    {% endfor %}
</table>
{% else %}
    <p>No polls are available.</p>
{% endif %}
</body>
</html>

视图方面的更改只涉及repathView,具体如下。

def repathView(request, number):
    latest_question_list = Question.objects.order_by('-pub_date')
    template = loader.get_template('pollsTemp/repathView.html')
    context = {
        'latest_question_list': latest_question_list,
    }
    return HttpResponse(template.render(context, request))

模板的加载是通过loader.get_template来完成的。
正如之前所述,它会在应用程序下的templates目录下查找模板,因此在这里需要指定的是从templates目录下的路径名。

传递给模板的变量需以字典类型传递。

最后,返回template.render(包含变量的字典, request)作为HttpResponse的参数。

像往常一样

运行以下命令:python manage.py runserver

启动服务器,并访问http://127.0.0.1:8000/polls/test1。

在比较之前,请先访问 http://127.0.0.1:8000/polls/tes1 来查看和 subView() 函数生成的结果是否相同。

快捷方式:呈现()

使用render()函数同时执行在视图中读取和发送模板的操作,来重写视图。

from django.shortcuts import render

def repathView(request, number):
    latest_question_list = Question.objects.order_by('-pub_date')
    context = {
        'latest_question_list': latest_question_list,
    }
    return render(request, 'pollsTemp/repathView.html', context)

这是最终的源代码。

展示到目前为止所修改的文件内容

"""
Django settings for demoDjango project.

Generated by 'django-admin startproject' using Django 2.0.2.

For more information on this file, see
https://docs.djangoproject.com/en/2.0/topics/settings/

For the full list of settings and their values, see
https://docs.djangoproject.com/en/2.0/ref/settings/
"""

import os

# Build paths inside the project like this: os.path.join(BASE_DIR, ...)
BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))


# Quick-start development settings - unsuitable for production
# See https://docs.djangoproject.com/en/2.0/howto/deployment/checklist/

# SECURITY WARNING: keep the secret key used in production secret!
SECRET_KEY = 'w2)3xu0&^)c)oa9s@-y#hg8&e=sy#08@6n656yzdvg6xgnt6_#'

# SECURITY WARNING: don't run with debug turned on in production!
DEBUG = True

ALLOWED_HOSTS = []


# Application definition

INSTALLED_APPS = [
    'polls.apps.PollsConfig',
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
]

MIDDLEWARE = [
    'django.middleware.security.SecurityMiddleware',
    'django.contrib.sessions.middleware.SessionMiddleware',
    'django.middleware.common.CommonMiddleware',
    'django.middleware.csrf.CsrfViewMiddleware',
    'django.contrib.auth.middleware.AuthenticationMiddleware',
    'django.contrib.messages.middleware.MessageMiddleware',
    'django.middleware.clickjacking.XFrameOptionsMiddleware',
]

ROOT_URLCONF = 'demoDjango.urls'

TEMPLATES = [
    {
        'BACKEND': 'django.template.backends.django.DjangoTemplates',
        'DIRS': [],
        'APP_DIRS': True,
        'OPTIONS': {
            'context_processors': [
                'django.template.context_processors.debug',
                'django.template.context_processors.request',
                'django.contrib.auth.context_processors.auth',
                'django.contrib.messages.context_processors.messages',
            ],
        },
    },
]

WSGI_APPLICATION = 'demoDjango.wsgi.application'


# Database
# https://docs.djangoproject.com/en/2.0/ref/settings/#databases

DATABASES = {
    'default': {
        'ENGINE': 'django.db.backends.sqlite3',
        'NAME': os.path.join(BASE_DIR, 'db.sqlite3'),
    }
}


# Password validation
# https://docs.djangoproject.com/en/2.0/ref/settings/#auth-password-validators

AUTH_PASSWORD_VALIDATORS = [
    {
        'NAME': 'django.contrib.auth.password_validation.UserAttributeSimilarityValidator',
    },
    {
        'NAME': 'django.contrib.auth.password_validation.MinimumLengthValidator',
    },
    {
        'NAME': 'django.contrib.auth.password_validation.CommonPasswordValidator',
    },
    {
        'NAME': 'django.contrib.auth.password_validation.NumericPasswordValidator',
    },
]


# Internationalization
# https://docs.djangoproject.com/en/2.0/topics/i18n/

LANGUAGE_CODE = 'ja'

TIME_ZONE = 'Asia/Tokyo'

USE_I18N = True

USE_L10N = True

USE_TZ = True


# Static files (CSS, JavaScript, Images)
# https://docs.djangoproject.com/en/2.0/howto/static-files/

STATIC_URL = '/static/'
"""demoDjango URL Configuration

The `urlpatterns` list routes URLs to views. For more information please see:
    https://docs.djangoproject.com/en/2.0/topics/http/urls/
Examples:
Function views
    1. Add an import:  from my_app import views
    2. Add a URL to urlpatterns:  path('', views.home, name='home')
Class-based views
    1. Add an import:  from other_app.views import Home
    2. Add a URL to urlpatterns:  path('', Home.as_view(), name='home')
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'))
"""
from django.contrib import admin
from django.urls import path, include

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

from .models import Question, Choice

admin.site.register(Question)
admin.site.register(Choice)
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)
from django.urls import path,re_path

from . import views
import re

urlpatterns = [
    path('', views.index, name='index'),
    path('<int:number>', views.subView, name='suburl'),
    re_path('test(?P<number>[0-9])', views.repathView, name='re_path_url'),
]
from django.http import HttpResponse
from pytz import timezone

from .models import Question
from django.shortcuts import render

def index(request):
    latest_question_list = Question.objects.order_by('-pub_date')[:5]
    output = ', '.join([q.question_text for q in latest_question_list])
    return HttpResponse(output)

    # Leave the rest of the views (detail, results, vote) unchanged

def subView(request, number):
    outStr1 ="<html>"\
            "<head>"\
            "<title>Question DBの中味を表示する</title>"\
            "</head>"\
            "<body>"\
            "<table>"\

    outStr2 ="</table>"\
             "</body>"\
             "</html>"

    latest_question_list = Question.objects.order_by('-pub_date')
    outList = ""

    for q in latest_question_list:
        outList += "<tr><td>" + q.pub_date.astimezone(timezone('Asia/Tokyo')).strftime( '%Y/%m/%d %H:%M:%S') +"</td><td>" + q.question_text + "</td></tr>"
    output = outStr1+ outList + outStr2
    return HttpResponse(output)

def repathView(request, number):
    latest_question_list = Question.objects.order_by('-pub_date')
    context = {
        'latest_question_list': latest_question_list,
    }
    return render(request, 'pollsTemp/repathView.html', context)
<!DOCTYPE html>
<html lang="ja">
<head>
    <meta charset="UTF-8">
    <title>Question DBの中味を表示する</title>
</head>
<body>

{% if latest_question_list %}
<table>
    {% for q in latest_question_list %}
    <tr>
        <td>{{q.pub_date | date:"Y/m/d " }}{{q.pub_date | time:"H:i:s" }}</td>
        <td>{{ q.question_text }}</td>
    </tr>
    {% endfor %}
</table>
{% else %}
    <p>No polls are available.</p>
{% endif %}
</body>
</html>
bannerAds