模型的建立
python manage.py startapp blog./blog/models.pyBlogArticles
通过这个类我们可以创建一个专门用来保存博客文章的数据库表,代码如下:
from django.db import modelsfrom django.utils import timezone # 新增from django.contrib.auth.models import User # 新增# Create your models here.# 编写博客的数据类型类BlogArticlesclass BlogArticles(models.Model): # 字段title的属性为CharField()类型,并且以参数max_length=300说明字段的最大数量
title = models.CharField(max_length=300) # 一个用户对应多篇文章,级联删除
author = models.ForeignKey(User, on_delete=models.CASCADE, related_name="blog_posts")
body = models.TextField()
publish = models.DateTimeField(default=timezone.now) class Meta:
ordering = ("-publish", ) # 规定了BlogArticles实例对象的显示顺序,即按照publish字段值的倒序显示
def __str__(self): return self.title
登录后复制
相信经过上一篇文章的介绍,这里很多面孔你都很熟悉,现在对他们进行简单介绍:
- timezone 模块,我们未来记录文章发布时间,所以导入了这个模块。但因为 Django 默认开启时区且为 UTC,所以需要在 settings 中进行如下设置:
TIME_ZONE = 'Asia/Shanghai' # 设置东八区时间# TIME_ZONE = 'UTC'USE_TZ = False
登录后复制
Userdjango.contrib.authINSTALL_APPSForeignKey()on_delete=models.CASCADErelated_name="blog_posts"ordering = ("-publish", )__str__
settings.pyINSTALLED_APPS
BlogArticles
数据迁移
manage.py
python manage.py makemigrations
登录后复制
然后你将看到如下输出内容:
Migrations for 'blog':
blog/migrations/0001_initial.py
- Create model BlogArticles
登录后复制
/blog/migrations0001_initial.py
迁移文件将被翻译成 SQL 语句。执行下面命令:
python manage.py sqlmigrate blog 0001
登录后复制
然后可以看到如下输出内容:
BEGIN;
--
-- Create model BlogArticles
--
CREATE TABLE "blog_blogarticles" ("id" integer NOT NULL PRIMARY KEY AUTOINCREMENT, "title" varchar(300) NOT NULL, "body" text NOT NULL, "publish" datetime NOT NULL, "author_id" integer NOT NULL REFERENCES "auth_user" ("id") DEFERRABLE INITIALLY DEFERRED);
CREATE INDEX "blog_blogarticles_author_id_ed798e23" ON "blog_blogarticles" ("author_id");
COMMIT;
登录后复制
最后我们将生产的迁移文件应用到数据库中:
python manage.py migrate
登录后复制
当看到
Operations to perform:
Apply all migrations: admin, auth, blog, contenttypes, sessions
Running migrations:
...
Applying sessions.001_initial... OK
登录后复制
Applying sessions.001_initial... OK
Django 自带一个数据库 SQLite,需要注意的是 SQLite 是一个产品级数据库。SQLite 被许多公司用于成千上万的产品,如所有 Android 和 iOS 设备,主流的 Web 浏览器,Windows 10,MacOS 等。 但这不适合所有情况。SQLite 不能与 MySQL,PostgreSQL 或 Oracle 等数据库进行比较。大容量的网站,密集型写入的应用程序,大的数据集,高并发性的应用使用 SQLite 最终都会导致问题。
SQLite 很轻量级,但是并不影响我们用来学习,接下来我们将利用命令行工具来对模型进行操作,也就是我们学习过数据库的基本操作——增、删、改、查。
模型操作
上面的内容中建立了一个博客文章的模型,然后通过数据迁移建立我们对应中的数据库表,加下来基于上述模型进行一些增删改成的操作,这小部分内容可能需要大家对数据库操作有一点熟悉。相信大家都是抱着学习 Django 的目的来看这一套入门笔记,我就不敢再各位数据库大佬们面前班门弄斧了,直接进入本章学习了。
SQLite 的启动
sqlite3 db.sqlite3
如果我们输入命令,看到如下信息,说明进入成功:
~/DjangoProject/myblog $ sqlite3 db.sqlite3
SQLite version 3.29.0 2019-07-10 17:32:03
Enter ".help" for usage hints.
sqlite>
登录后复制
.tablesblog_blogarticles
~/DjangoProject/myblog $ sqlite3 db.sqlite3
SQLite version 3.29.0 2019-07-10 17:32:03
Enter ".help" for usage hints.
sqlite> .tables
auth_group blog_blogarticles
auth_group_permissions django_admin_log
auth_permission django_content_type
auth_user django_migrations
auth_user_groups django_session
auth_user_user_permissions
sqlite>
登录后复制
pragma table_info(blog_blogarticles);
sqlite> .header on
sqlite> pragma table_info(blog_blogarticles);
cid|name |type |notnull|dflt_value|pk
0| id |integer |1 | |1
1|title |varchar(300)|1 | |0
2|body |text |1 | |0
3|publish |datetime |1 | |0
4|author_id|integer |1 | |0
sqlite>
登录后复制
PRAGMA
我们可以大致查看上面的表结构,cid 是指列 id,name 是指列名,type 是指列类型,notnull 非空,值为 1 表示 True,dflt_value 是指 default 默认值(这一列没有值,说明设置设置默认值),pk 是指 primary_key 主键。
大家可以跟我们在前一章中的数据模型 BlogArticles 所规定的字段和属性进行对比,是不是刚好我们利用数据迁移成功将数据模型转化为数据库表。
创建超级管理员
python manage.py createsuperuser
~/DjangoProject/myblog $ python manage.py createsuperuser
Username (leave blank to use 'yuzhou_1su'): zoeu
Email address: test@test.com
Password:
Password (again):
Superuser created successfully.
登录后复制
http://127.0.0.1:8000/admin/
输入刚才创建的超级管理员的用户名和密码就可以进入系统,如图:
./blog/admin.py
from django.contrib import admin# 新增,将BlogArticles类引入到当前环境中from .models import BlogArticles
# 将BlogArticles注册到admin中admin.site.register(BlogArticles)
登录后复制
刷新页面,我们可以得到如下的页面:
超级管理员界面先放在这,我们回到模型操作。
模型 API 测试
./blog/models.py
manage.py
python3 manage.py shell
登录后复制
Python 3.7.3 (v3.7.3:ef4ec6ed12, Mar 25 2019, 16:52:21)
[Clang 6.0 (clang-600.0.57)] on darwin
Type "help", "copyright", "credits" or "license" for more information.
(InteractiveConsole)
>>>
登录后复制
请特别注意我们进入 shell 的方式,不是直接在命令行中敲 python3,虽然这与直接输入 python 指令来调用交互式控制台是非常相似。
sys.path
让我们从导入 BlogArticles 类开始:下面就可以开始我们对数据库的增、删、改、查等操作。
$ python3 manage.py shell
Python 3.7.3 (v3.7.3:ef4ec6ed12, Mar 25 2019, 16:52:21)
[Clang 6.0 (clang-600.0.57)] on darwinType "help", "copyright", "credits" or "license" for more information.
(InteractiveConsole)>>> from django.contrib.auth.models import User>>> from blog.models import BlogArticles>>> admin = User.objects.get(username="zoue")>>> admin.username'zoeu'>>> admin.id1>>> admin.password'pbkdf2_sha256$150000$b9j0ZKBVZSo1$l+fEIiKIaS6u1mhjMPOX1qR0xMOaemnRJIwiE2lNn60='>>> admin.email'test@test.com'>>> type(admin)
<class 'django.contrib.auth.models.User'>
登录后复制
以上是对用户的查询操作是不是刚好是我们上一节中在创建管理员的内容,恭喜你!
接下来,我们对博客文章进行操作,要创建新的 BlogArticle 对象,我们可以执行以下操作:
- 增
>>> BlogAriticle01 = BlogArticles(title ='DjangoNotes_Day01', author=zoue, body='Django是啥?');
登录后复制
为了将这个对象保存在数据库中,我们必须调用 save 方法:
>>> BlogAriticle01.save()
登录后复制
save
>>> BlogAriticle01.id4
登录后复制
因为我自己在之前创建过其它文章,所以这个 id 值为 4,如果你是按照本入门笔记一步步操作过来,id 值应该为 1。
当然,还可以查看其它属性,这里统一将命令敲出来:
>>> BlogAriticle01.title'DjangoNotes_Day01'>>> BlogAriticle01.author
<User: admin>>>> BlogAriticle01.body'Django是啥?'>>> BlogAriticle01.publish
datetime.datetime(2019, 9, 30, 19, 56, 58, 735676)>>>
登录后复制
每个 Django 模型都带有一个特殊的属性; 我们称之为模型管理器(Model Manager)。你可以通过属性 objects 来访问这个管理器,它主要用于数据库操作。例如,我们可以使用它来直接创建一个新的 Board 对象:
>>> BlogArticle02 = BlogArticles.objects.create(title='Python', author=admin, body='Head First to Python.')>>> BlogArticle02.id5
登录后复制
- 改
要更新一个值,我们可以利用如下操作:
>>> BlogAriticle01.body = 'I love Django, 但是我太难了'>>> BlogAriticle01.body'I love Django, 但是我太难了'
登录后复制
- 查
>>> blogs = BlogArticles.objects.all()>>> blogs
<QuerySet [<BlogArticles: Python>, <BlogArticles: DjangoNotes_Day01>, <BlogArticles: right here waiting>, <BlogArticles: Yesterday once more>, <BlogArticles: You Raise me up>]>
登录后复制
结果是一个 QuerySet,我们可以将这个 QuerySet 看作一个列表。假设我们想遍历它并打印每个模块的标题。
>>> for blog in blogs:... print(blog.title)... Python
DjangoNotes_Day01
right here waiting
Yesterday once more
You Raise me up>>>
登录后复制
同样,我们可以使用模型的 管理器(Manager) 来查询数据库并返回单个对象。为此,我们要使用 get 方法:
>>> BlogArticles.objects.get(id=5)
<BlogArticles: Python>
登录后复制
- 删
>>> BlogArticles.objects.get(id=5).delete()
(1, {'blog.BlogArticles': 1})
登录后复制
删除后再重新查看一下 QuerySet,发现没有了<BlogArticles: Python>,说明删除成功。
>>> blogs = BlogArticles.objects.all()>>> blogs
<QuerySet [<BlogArticles: DjangoNotes_Day01>, <BlogArticles: right here waiting>, <BlogArticles: Yesterday once more>, <BlogArticles: You Raise me up>]>
登录后复制
BlogArticles.objects.filter(id=5).delete()
总结
下面是我们在本节中关于模型学到的方法和操作,使用 BlogArticles 模型作为参考。大写的 BlogArticles 指的是类,BlogArticles01 指 BlogArticles 的一个实例(或对象):
BlogAriticle01 = BlogArticles()BlogAriticle01.save()BlogArticle02 = BlogArticles.objects.create(title='...', author=..., body='...')BlogArticles.objects.all()BlogArticles.objects.get(id=5)BlogArticles.objects.get(id=5).delete()