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`

在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>