数据库配置

打开 mysite/settings.py ,这是个包含了 Django 项目设置的 Python 模块。
通常,这个配置文件使用 SQLite 作为默认数据库。Python 内置 SQLite,所以你无需安装额外东西来使用它。当你开始一个真正的项目时,你可能更倾向使用一个更具扩展性的数据库,例如 PostgreSQL,避免中途切换数据库这个令人头疼的问题。
如果你想使用其他数据库,你需要安装合适的 database bindings ,然后改变设置文件中 DATABASES ‘default’ 项目中的一些键值:

  • ENGINE — 可选值有 ‘django.db.backends.sqlite3’,’django.db.backends.postgresql’,’django.db.backends.mysql’,或 ‘django.db.backends.oracle’。
  • NAME — 数据库的名称。如果你使用 SQLite,数据库将是你电脑上的一个文件,在这种情况下,NAME应该是此文件完整的绝对路径,包括文件名。默认值 BASE_DIR / ‘db.sqlite3’ 将把数据库文件储存在项目的根目录。

如果你不使用 SQLite,则必须添加一些额外设置,比如 USER 、 PASSWORD 、 HOST 等等。编辑 mysite/settings.py 文件前,先设置 TIME_ZONE为你自己时区。

Django 的自带应用

此外,关注一下文件头部的 INSTALLED_APPS设置项。这里包括了会在你项目中启用的所有 Django 应用。应用能在多个项目中使用,你也可以打包并且发布应用,让别人使用它们。
通常, INSTALLED_APPS默认包括了以下 Django 的自带应用:

  • django.contrib.admin — 管理员站点, 你很快就会使用它。
  • django.contrib.auth — 认证授权系统。
  • django.contrib.contenttypes — 内容类型框架。
  • django.contrib.sessions — 会话框架。
  • django.contrib.messages — 消息框架。
  • django.contrib.staticfiles — 管理静态文件的框架。

这些应用被默认启用是为了给常规项目提供方便。
默认开启的某些应用需要至少一个数据表,所以在使用他们之前需要在数据库中创建一些表。请执行以下命令:
…> py manage.py migrate
这个 migrate命令查看 INSTALLED_APPS 配置,并根据 mysite/settings.py 文件中的数据库配置和随应用提供的数据库迁移文件(我们将在后面介绍这些),创建任何必要的数据库表。你会看到它应用的每一个迁移都有一个消息。如果你有兴趣,运行你的数据库的命令行客户端,输入 \dt (PostgreSQL), SHOW TABLES;(MariaDB,MySQL), .tables (SQLite)或 SELECT TABLE_NAME FROM USER_TABLES; (Oracle)来显示 Django 创建的表。

创建模型

在 Django 里写一个数据库驱动的 Web 应用的第一步是定义模型 - 也就是数据库结构设计和附加的其它元数据。
在这个投票应用中,需要创建两个模型:问题 Question 和选项 Choice。Question 模型包括问题描述和发布时间。Choice 模型有两个字段,选项描述和当前得票数。每个选项属于一个问题。
这些概念可以通过一个 Python 类来描述。按照下面的例子来编辑 polls/models.py 文件:

  1. # Create your models here.
  2. from django.db import models
  3. class Question(models.Model):
  4. question_text = models.CharField(max_length=200)
  5. pub_date = models.DateTimeField('date published')
  6. class Choice(models.Model):
  7. question = models.ForeignKey(Question, on_delete=models.CASCADE) # 定义外键关系
  8. choice_text = models.CharField(max_length=200)
  9. votes = models.IntegerField(default=0)

每个模型被表示为 django.db.models.Model 类的子类。每个模型有许多类变量,它们都表示模型里的一个数据库字段。
每个字段都是 Field类的实例 - 比如,字符字段被表示为 CharField,日期时间字段被表示为 DateTimeField。这将告诉 Django 每个字段要处理的数据类型。
每个 Field 类实例变量的名字(例如 question_text 或 pub_date )也是字段名,所以最好使用对机器友好的格式。你将会在 Python 代码里使用它们,而数据库会将它们作为列名。
定义某些 Field 类实例需要参数。例如 CharField 需要一个 max_length 参数。这个参数的用处不止于用来定义数据库结构,也用于验证数据。
Field 也能够接收多个可选参数;在上面的例子中:我们将 votes 的 default 也就是默认值,设为0。
注意在最后,我们使用 ForeignKey 定义了一个关系。这将告诉 Django,每个 Choice 对象都关联到一个 Question 对象。Django 支持所有常用的数据库关系:多对一、多对多和一对一。

激活模型

上面的一小段用于创建模型的代码给了 Django 很多信息,通过这些信息,Django 可以:

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

但是首先得把 polls 应用安装到我们的项目里。

为了在我们的工程中包含这个应用,我们需要在配置类 INSTALLED_APPS 中添加设置。因为 PollsConfig 类写在文件 polls/apps.py 中,所以它的点式路径是 ‘polls.apps.PollsConfig’。在文件 mysite/settings.py 中 INSTALLED_APPS 子项添加点式路径后,它看起来像这样:
image.png
现在你的 Django 项目会包含 polls 应用。接着运行下面的命令:
…> py manage.py makemigrations polls
你将会看到类似于下面这样的输出:
image.png
通过运行 makemigrations 命令,Django 会检测你对模型文件的修改(在这种情况下,你已经取得了新的),并且把修改的部分储存为一次迁移。
迁移是 Django 对于模型定义(也就是你的数据库结构)的变化的储存形式 - 它们其实也只是一些你磁盘上的文件。如果你想的话,你可以阅读一下你模型的迁移数据,它被储存在 polls/migrations/0001_initial.py 里。别担心,你不需要每次都阅读迁移文件,但是它们被设计成人类可读的形式,这是为了便于你手动调整 Django 的修改方式。
Django 有一个自动执行数据库迁移并同步管理你的数据库结构的命令 - 这个命令是 migrate,我们马上就会接触它 - 但是首先,让我们看看迁移命令会执行哪些 SQL 语句。sqlmigrate 命令接收一个迁移的名称,然后返回对应的 SQL:
…> py manage.py sqlmigrate polls 0001
你将会看到类似下面这样的输出(我把输出重组成了人类可读的格式):

  1. BEGIN;
  2. --
  3. -- Create model Question
  4. --
  5. CREATE TABLE "polls_question" (
  6. "id" serial NOT NULL PRIMARY KEY,
  7. "question_text" varchar(200) NOT NULL,
  8. "pub_date" timestamp with time zone NOT NULL
  9. );
  10. --
  11. -- Create model Choice
  12. --
  13. CREATE TABLE "polls_choice" (
  14. "id" serial NOT NULL PRIMARY KEY,
  15. "choice_text" varchar(200) NOT NULL,
  16. "votes" integer NOT NULL,
  17. "question_id" integer NOT NULL
  18. );
  19. ALTER TABLE "polls_choice"
  20. ADD CONSTRAINT "polls_choice_question_id_c5b4b260_fk_polls_question_id"
  21. FOREIGN KEY ("question_id")
  22. REFERENCES "polls_question" ("id")
  23. DEFERRABLE INITIALLY DEFERRED;
  24. CREATE INDEX "polls_choice_question_id_c5b4b260" ON "polls_choice" ("question_id");
  25. COMMIT;

请注意以下几点:

  • 输出的内容和你使用的数据库有关,上面的输出示例使用的是 PostgreSQL。
  • 数据库的表名是由应用名(polls)和模型名的小写形式( question 和 choice)连接而来。(如果需要,你可以自定义此行为。)主键(IDs)会被自动创建。(当然,你也可以自定义。)
  • 默认的,Django 会在外键字段名后追加字符串 “_id” 。(同样,这也可以自定义。)
  • 外键关系由 FOREIGN KEY 生成。你不用关心 DEFERRABLE部分,它只是告诉 PostgreSQL,请在事务全都执行完之后再创建外键关系。
  • 生成的 SQL 语句是为你所用的数据库定制的,所以那些和数据库有关的字段类型,比如 auto_increment(MySQL)、 serial (PostgreSQL)和 integer primary key autoincrement (SQLite),Django 会帮你自动处理。那些和引号相关的事情 - 例如,是使用单引号还是双引号 - 也一样会被自动处理。
  • 这个 sqlmigrate命令并没有真正在你的数据库中的执行迁移 - 相反,它只是把命令输出到屏幕上,让你看看 Django 认为需要执行哪些 SQL 语句。这在你想看看 Django 到底准备做什么,或者当你是数据库管理员,需要写脚本来批量处理数据库时会很有用。

如果你感兴趣,你也可以试试运行 python manage.py check ;这个命令帮助你检查项目中的问题,并且在检查过程中不会对数据库进行任何操作。
现在,再次运行 migrate命令,在数据库里创建新定义的模型的数据表:
…> py manage.py migrate
image.png
这个 migrate命令选中所有还没有执行过的迁移(Django 通过在数据库中创建一个特殊的表 django_migrations来跟踪执行过哪些迁移)并应用在数据库上 - 也就是将你对模型的更改同步到数据库结构上。
迁移是非常强大的功能,它能让你在开发过程中持续的改变数据库结构而不需要重新删除和创建表 - 它专注于使数据库平滑升级而不会丢失数据。我们会在后面的教程中更加深入的学习这部分内容.

修改模型的步骤总结

现在,你只需要记住,改变模型需要这三步:

  • 编辑 models.py 文件,改变模型。
  • 运行 python manage.py makemigrations 为模型的改变生成迁移文件。
  • 运行 python manage.py migrate 来应用数据库迁移。

数据库迁移被分解成生成和应用两个命令是为了让你能够在代码控制系统上提交迁移数据并使其能在多个应用里使用;这不仅仅会让开发更加简单,也给别的开发者和生产环境中的使用带来方便。

初试API

现在让我们进入交互式 Python 命令行,尝试一下 Django 为你创建的各种 API。通过以下命令打开 Python 命令行:
…> py manage.py shell
我们使用这个命令而不是简单的使用“python”是因为 manage.py 会设置 DJANGO_SETTINGS_MODULE环境变量,这个变量会让 Django 根据 mysite/settings.py 文件来设置 Python 包的导入路径。
当你成功进入命令行后,来试试 数据库 API 吧:

  1. >>> from polls.models import Choice, Question # Import the model classes we just wrote.
  2. # No questions are in the system yet.
  3. >>> Question.objects.all()
  4. <QuerySet []>
  5. # Create a new Question.
  6. # Support for time zones is enabled in the default settings file, so
  7. # Django expects a datetime with tzinfo for pub_date. Use timezone.now()
  8. # instead of datetime.datetime.now() and it will do the right thing.
  9. >>> from django.utils import timezone
  10. >>> q = Question(question_text="What's new?", pub_date=timezone.now())
  11. # Save the object into the database. You have to call save() explicitly.
  12. >>> q.save()
  13. # Now it has an ID.
  14. >>> q.id
  15. 1
  16. # Access model field values via Python attributes.
  17. >>> q.question_text
  18. "What's new?"
  19. >>> q.pub_date
  20. datetime.datetime(2012, 2, 26, 13, 0, 0, 775217, tzinfo=<UTC>)
  21. # Change values by changing the attributes, then calling save().
  22. >>> q.question_text = "What's up?"
  23. >>> q.save()
  24. # objects.all() displays all the questions in the database.
  25. >>> Question.objects.all()
  26. <QuerySet [<Question: Question object (1)>]>

对于我们了解这个对象的细节没什么帮助。让我们通过编辑 Question 模型的代码(位于 polls/models.py 中)来修复这个问题。给 Question和 Choice增加 str() 方法。

  1. from django.db import models
  2. class Question(models.Model):
  3. # ...
  4. def __str__(self):
  5. return self.question_text
  6. class Choice(models.Model):
  7. # ...
  8. def __str__(self):
  9. return self.choice_text

给模型增加 str() 方法是很重要的,这不仅仅能给你在命令行里使用带来方便,Django 自动生成的 admin 里也使用这个方法来表示对象。
让我们再为此模型添加一个自定义方法:

  1. import datetime
  2. from django.db import models
  3. from django.utils import timezone
  4. class Question(models.Model):
  5. # ...
  6. def was_published_recently(self):
  7. return self.pub_date >= timezone.now() - datetime.timedelta(days=1)

新加入的 import datetime 和 from django.utils import timezone 分别导入了 Python 的标准 datetime模块和 Django 中和时区相关的 django.utils.timezone 工具模块。
保存文件然后通过 python manage.py shell命令再次打开 Python 交互式命令行:

  1. >>> from polls.models import Choice, Question
  2. # Make sure our __str__() addition worked.
  3. >>> Question.objects.all()
  4. <QuerySet [<Question: What's up?>]>
  5. # Django provides a rich database lookup API that's entirely driven by
  6. # keyword arguments.
  7. >>> Question.objects.filter(id=1)
  8. <QuerySet [<Question: What's up?>]>
  9. >>> Question.objects.filter(question_text__startswith='What')
  10. <QuerySet [<Question: What's up?>]>
  11. # Get the question that was published this year.
  12. >>> from django.utils import timezone
  13. >>> current_year = timezone.now().year
  14. >>> Question.objects.get(pub_date__year=current_year)
  15. <Question: What's up?>
  16. # Request an ID that doesn't exist, this will raise an exception.
  17. >>> Question.objects.get(id=2)
  18. Traceback (most recent call last):
  19. ...
  20. DoesNotExist: Question matching query does not exist.
  21. # Lookup by a primary key is the most common case, so Django provides a
  22. # shortcut for primary-key exact lookups.
  23. # The following is identical to Question.objects.get(id=1).
  24. >>> Question.objects.get(pk=1)
  25. <Question: What's up?>
  26. # Make sure our custom method worked.
  27. >>> q = Question.objects.get(pk=1)
  28. >>> q.was_published_recently()
  29. True
  30. # Give the Question a couple of Choices. The create call constructs a new
  31. # Choice object, does the INSERT statement, adds the choice to the set
  32. # of available choices and returns the new Choice object. Django creates
  33. # a set to hold the "other side" of a ForeignKey relation
  34. # (e.g. a question's choice) which can be accessed via the API.
  35. >>> q = Question.objects.get(pk=1)
  36. # Display any choices from the related object set -- none so far.
  37. >>> q.choice_set.all()
  38. <QuerySet []>
  39. # Create three choices.
  40. >>> q.choice_set.create(choice_text='Not much', votes=0)
  41. <Choice: Not much>
  42. >>> q.choice_set.create(choice_text='The sky', votes=0)
  43. <Choice: The sky>
  44. >>> c = q.choice_set.create(choice_text='Just hacking again', votes=0)
  45. # Choice objects have API access to their related Question objects.
  46. >>> c.question
  47. <Question: What's up?>
  48. # And vice versa: Question objects get access to Choice objects.
  49. >>> q.choice_set.all()
  50. <QuerySet [<Choice: Not much>, <Choice: The sky>, <Choice: Just hacking again>]>
  51. >>> q.choice_set.count()
  52. 3
  53. # The API automatically follows relationships as far as you need.
  54. # Use double underscores to separate relationships.
  55. # This works as many levels deep as you want; there's no limit.
  56. # Find all Choices for any question whose pub_date is in this year
  57. # (reusing the 'current_year' variable we created above).
  58. >>> Choice.objects.filter(question__pub_date__year=current_year)
  59. <QuerySet [<Choice: Not much>, <Choice: The sky>, <Choice: Just hacking again>]>
  60. # Let's delete one of the choices. Use delete() for that.
  61. >>> c = q.choice_set.filter(choice_text__startswith='Just hacking')
  62. >>> c.delete()

介绍Django管理页面

为你的员工或客户生成一个用户添加,修改和删除内容的后台是一项缺乏创造性和乏味的工作。因此,Django 全自动地根据模型创建后台界面。
Django 产生于一个公众页面和内容发布者页面完全分离的新闻类站点的开发过程中。站点管理人员使用管理系统来添加新闻、事件和体育时讯等,这些添加的内容被显示在公众页面上。Django 通过为站点管理人员创建统一的内容编辑界面解决了这个问题。
管理界面不是为了网站的访问者,而是为管理者准备的。

创建一个管理员账号

首先,我们得创建一个能登录管理页面的用户。请运行下面的命令:
…> py manage.py createsuperuser
键入你想要使用的用户名,然后按下回车键:
Username: admin
然后提示你输入想要使用的邮件地址:
Email address: admin@example.com
最后一步是输入密码。你会被要求输入两次密码,第二次的目的是为了确认第一次输入的确实是你想要的密码。
Password: ** Password (again): *** Superuser created successfully.

启动开发服务器

Django 的管理界面默认就是启用的。让我们启动开发服务器,看看它到底是什么样的。
如果开发服务器未启动,用以下命令启动它:
…> py manage.py runserver
现在,打开浏览器,转到你本地域名的 “/admin/” 目录, — 比如 http://127.0.0.1:8000/admin/ 。你应该会看见管理员登录界面:

image.png

因为翻译功能默认是开启的,如果你设置了 LANGUAGE_CODE,登录界面将显示你设置的语言(如果 Django 有相应的翻译)。

进入管理站点页面

现在,试着使用你在上一步中创建的超级用户来登录。然后你将会看到 Django 管理页面的索引页:

image.png

你将会看到几种可编辑的内容:组和用户。它们是由 django.contrib.auth 提供的,这是 Django 开发的认证框架。

向管理页面中加入投票应用

但是我们的投票应用在哪呢?它没在索引页面里显示。
只需要再做一件事:我们得告诉管理,问题 Question 对象需要一个后台接口。打开 polls/admin.py 文件,把它编辑成下面这样:

  1. from django.contrib import admin
  2. from .models import Question
  3. admin.site.register(Question)

体验便捷的管理功能

现在我们向管理页面注册了问题 Question类。Django 知道它应该被显示在索引页里:

image.png

点击 “Questions” 。现在看到是问题 “Questions” 对象的列表 “change list” 。这个界面会显示所有数据库里的问题 Question 对象,你可以选择一个来修改。这里现在有我们在上一部分中创建的 “What’s up?” 问题。

image.png

点击 “What’s up?” 来编辑这个问题(Question)对象:

image.png

注意事项:

  • 这个表单是从问题 Question 模型中自动生成的
  • 不同的字段类型(日期时间字段 DateTimeField 、字符字段 CharField)会生成对应的 HTML 输入控件。每个类型的字段都知道它们该如何在管理页面里显示自己。
  • 每个日期时间字段 DateTimeField 都有 JavaScript 写的快捷按钮。日期有转到今天(Today)的快捷按钮和一个弹出式日历界面。时间有设为现在(Now)的快捷按钮和一个列出常用时间的方便的弹出式列表。

页面的底部提供了几个选项:

  • 保存(Save) - 保存改变,然后返回对象列表。
  • 保存并继续编辑(Save and continue editing) - 保存改变,然后重新载入当前对象的修改界面。
  • 保存并新增(Save and add another) - 保存改变,然后添加一个新的空对象并载入修改界面。
  • 删除(Delete) - 显示一个确认删除页面。

如果显示的 “发布日期(Date Published)” 和你在 教程 1 里创建它们的时间不一致,这意味着你可能没有正确的设置 TIME_ZONE。改变设置,然后重新载入页面看看是否显示了正确的值。
通过点击 “今天(Today)” 和 “现在(Now)” 按钮改变 “发布日期(Date Published)”。然后点击 “保存并继续编辑(Save and add another)”按钮。然后点击右上角的 “历史(History)”按钮。你会看到一个列出了所有通过 Django 管理页面对当前对象进行的改变的页面,其中列出了时间戳和进行修改操作的用户名:
image.png