Django06模型和数据库

首先,让我们看看Rango的需求.下面列出了Rango数据关键的几个需求.

  • Rango实际上是一个网页目录 - 一个包含其他我站链接的网站
  • 有许多不同网站的目录,每个目录中包含许多链接.我们在第二章假设1对多关系 看下面的ER图.
  • 一个目录要有名字,访问数和链接.
  • 一个页面要有目录,标题,URL和一些视图

1. 告诉Django你的数据库

在你创造任何模型之前都要对你的数据库进行设置
Django会自动在settings.py里添加一个叫做DATABASES的字典.它包含如下.

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

能看到默认用SQLite3作为后端数据库.SQLite是个轻量级的数据库对我们开发很有用.我们仅仅需要设置DATABASE_PATH里的NAME键值对.其他引擎需要USER,PASSWORD,HOST和PORT等关键字

2. 创建模型

让我们为Rango创建两个数据模型.

在rango/models.py里,我们定义两个继承自djnago.db.models.Model的类.这两个类分别定义目录和页面.定义Category和Page如下

from django.db import models

# Create your models here.

class Category(models.Model):
    name = models.CharField(max_length=128, unique=True)
    def __str__(self):
        return self.name

class Page(models.Model):
    category = models.ForeignKey(Category)
    title = models.CharField(max_length=128)
    url = models.URLField()
    views = models.IntegerField(default=0)
    def __str__(self):
        return self.title

当你定义了一个模型,你需要制定可选参数的属性以及相关的类型列表.Django提供了许多内建字段.一些常用的如下.

  • CharField,存储字符数据的字段(例如字符串).max_length提供了最大长度.
  • URLField,和CharField一样,但是它存储资源的URL.你也可以使用max_length参数.
  • 'IntegerField',存储整数.
  • DateField,存储Python的datetime.date.

每个字段都有一个unique属性.如果设置为True,那么在整个数据库模型里它的字段里的值必须是唯一的.例如,我们上面定义的Category模型.name字段被设置为unique - 所以每一个目录的名字都必须是唯一的
如果你想把这个字段作为数据库的关键字会非常有用.你可以为每个字段设置一个默认值(default='value'),也可以设置成NULL(null=True).
Django也提供了连接模型/表的简单机制.这个机制封装在3个字段里,如下.

  • ForeignKey, 创建1对多关系的字段类型.
  • OneToOneField,定义一个严格的1对1关系字段类型.
  • ManyToManyFeild,当以多对多关系字段类型.
    从上面我们的例子,Page中category字段是ForeignKey类型.所以我们可以创建一个1对多关系的Category模型/表,这个Category会作为构造函数的一个参数.Django会自动的为每个模型表中创建ID字段.所以你不同为每个模型创建主键 - 它已经为你做好了!

3. 设置数据库并创建管理员

如果还没有设置数据库那么需要通过migrate命令来设立.

$ python manage.py migrate
Operations to perform:
  Apply all migrations: admin, contenttypes, auth, sessions
Running migrations:
  Applying contenttypes.0001_initial... OK
  Applying auth.0001_initial... OK
  Applying admin.0001_initial... OK
  Applying sessions.0001_initial... OK

你是否还记得在settings.py文件里设置的INSTALLED_APPS列表,它会初始化创建这些app的图表,例如auth,admin等等.在你的项目根目录里会创建db.sqlite文件.

  • 现在你可以创建一个管理员来管理数据库.
    $ python manage.py createsuperuser

4. 创建/上传模型/表

当你更改模型的时候,你需要通过makemigrations进行修改,所以对于rango,我们需要:

$ python manage.py makemigrations rango
Migrations for 'rango':
  0001_initial.py:
    - Create model Category
    - Create model Page

如果你检查rango/migrations文件,你会发现会创建一个叫做0001_initial.py的python脚本.如果想要SQL命令去执行迁移,那么可以用下面的命令python manage.py sqlmigrate 现在让我们应用这些迁移(基于创建数据库图表).

$ python manage.py migrate
Operations to perform:
  Apply all migrations: admin, rango, contenttypes, auth, sessions
Running migrations:
  Applying rango.0001_initial... OK

当你添加已存在的模型,你需要重复运行python manage.py makemigrations 命令然后再运行python manage.py migrate.

5. Shell

Django06模型和数据库_第1张图片
shell
# Import the Category model from the Rango application
>>> from rango.models import Category
# Show all the current categories
>>> print Category.objects.all()
[] # Returns an empty list (no categories have been defined!)
# Create a new category object, and save it to the database.
>>> c = Category(name="Test")
>>> c.save()
# Now list all the category objects stored once more.
>>> print Category.objects.all()
[] # We now have a category called 'test' saved in the database!
# Quit the Django shell.
>>> quit()

在例子中我们首先导入我们需要操作的模型.然后打印出存在的目录,在这里因为我们的图表是空所以输出也是空.然后创建并储存一个目录,打印.

6. 设置管理界面

Django最突出的一个特性就是它提供内建的网页管理界面,用来浏览和编辑存储在模型的数据,也可以与数据库图表交互.在settings.py文件里,注意到有一个默认安装的django.contib.adminapp,而且你的urls.py里也默认增加了admin/匹配.

开启Django服务:
$ python manage.py runserver
访问127.0.0.1:8000/admin.可以用先前设置管理员账户的用户名和密码来登录Django管理界面.管理界面只包含Groups和Users图表以我们需要让Django包含rango模块.所以打开rango/admin.py输入如下代码:

from django.contrib import admin
from rango.models import Category, Page
admin.site.register(Category)
admin.site.register(Page)

上面代码会为我们在管理界面注册模型.如果我们想要其他模型,可以在admin.stie.register()函数里传递模型作为参数.

完成之后重新访问127.0.0.1:8000/admin/,你想回看到如下图案.

Django06模型和数据库_第2张图片
效果图

7. 创建Population Script

往数据库里输入数据会非常麻烦.许多开发者会随机的往数据库里输入测试数据.如果你在一个小的开发团队里,每个人都得传点数据.最好是写一个脚本而不是每个人单独的上传数据,这样就可以避免垃圾数据的产生.所以我们需要为你的数据库创建 population script.这个脚本自动的为你的数据库生成测试数据.

我们需要在Django项目的根目录里创建population script(例如/tango_with_django_project/).创建populate_rango.py文件代码如下.

import os
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'tango_with_django_project.settings')

import django
django.setup()

from rango.models import Category, Page


def populate():
    python_cat = add_cat('Python')

    add_page(cat=python_cat,
        title="Official Python Tutorial",
        url="http://docs.python.org/2/tutorial/")

    add_page(cat=python_cat,
        title="How to Think like a Computer Scientist",
        url="http://www.greenteapress.com/thinkpython/")

    add_page(cat=python_cat,
        title="Learn Python in 10 Minutes",
        url="http://www.korokithakis.net/tutorials/python/")

    django_cat = add_cat("Django")

    add_page(cat=django_cat,
        title="Official Django Tutorial",
        url="https://docs.djangoproject.com/en/1.5/intro/tutorial01/")

    add_page(cat=django_cat,
        title="Django Rocks",
        url="http://www.djangorocks.com/")

    add_page(cat=django_cat,
        title="How to Tango with Django",
        url="http://www.tangowithdjango.com/")

    frame_cat = add_cat("Other Frameworks")

    add_page(cat=frame_cat,
        title="Bottle",
        url="http://bottlepy.org/docs/dev/")

    add_page(cat=frame_cat,
        title="Flask",
        url="http://flask.pocoo.org")

    # Print out what we have added to the user.
    for c in Category.objects.all():
        for p in Page.objects.filter(category=c):
            print ("- {0} - {1}".format(str(c), str(p)))

def add_page(cat, title, url, views=0):
    p = Page.objects.get_or_create(category=cat, title=title, url=url, views=views)[0]
    return p

def add_cat(name):
    c = Category.objects.get_or_create(name=name)[0]
    return c

# Start execution here!
if __name__ == '__main__':
    print ("Starting Rango population script...")
    populate()

$ python populate_rango.py执行脚本

Django06模型和数据库_第3张图片
效果图

8. 基本流程

Django06模型和数据库_第4张图片
流程图

9. 练习

Django06模型和数据库_第5张图片
练习

添加views和likes的代码:

Django06模型和数据库_第6张图片
代码

populate.py的修改

Django06模型和数据库_第7张图片
修改

定制管理界面

Django06模型和数据库_第8张图片
admin

最终效果

Django06模型和数据库_第9张图片
1
Django06模型和数据库_第10张图片
2

你可能感兴趣的:(Django06模型和数据库)