[关闭]
@breakerthb 2016-06-30T03:50:29.000000Z 字数 3463 阅读 1300

Django数据库模型

Django


Django 模型是与数据库相关的,与数据库相关的代码一般写在 models.py 中,Django 支持 sqlite3, MySQL, PostgreSQL等数据库,只需要在settings.py中配置即可,不用更改models.py中的代码,丰富的API极大的方便了使用。

创建项目、应用

$ django-admin.py startproject learn_models # 新建一个项目
	$ cd learn_models # 进入到该项目的文件夹
$ django-admin.py startapp people # 新建一个 people 应用(app)

在settings.py的 INSTALLED_APPS中添加应用。

INSTALLED_APPS = (
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',

    'people',
)

创建model

我们打开 people/models.py 文件,修改其中的代码如下:

from django.db import models

class Person(models.Model):
    name = models.CharField(max_length=30)
    age = models.IntegerField()

我们新建了一个Person类,继承自models.Model, 一个人有姓名和年龄。这里用到了两种Field,更多Field类型可以参考教程最后的链接。

同步数据库

$ python manage.py syncdb # 进入 manage.py 所在的那个文件夹下输入这个命令

注意:Django 1.7 及以上的版本需要用以下命令

$ python manage.py makemigrations
	$ python manage.py migrate

默认使用sqlite数据库。

管理数据库

    $ python manage.py shell 
    >>> from people.models import Person

我们用了一个 .objects.get() 方法查询出来符合条件的对象,但是大家注意到了没有,查询结果中显示,这里并没有显示出与tom的相关信息,如果用户多了就无法知道查询出来的到底是谁,查询结果是否正确

新修改一下 people/models.py

name 和 age 等字段中不能有 __(双下划线,因为在Django QuerySet API中有特殊含义(用于关系,包含,不区分大小写,以什么开头或结尾,日期的大于小于,正则等)

也不能有Python中的关键字,name 是合法的,student_name 也合法,但是student__name不合法,try, class, continue 也不合法,因为它是Python的关键字( import keyword; print(keyword.kwlist) 可以打出所有的关键字)

from django.db import models

class Person(models.Model):
    name = models.CharField(max_length=30)
    age = models.IntegerField()

    def __unicode__(self):
    # 在Python3中使用 def __str__(self)
        return self.name

重新查询:

新建对象的方法

Person.objects.create(name=name,age=age)

p = Person(name="WZ", age=23)

p.save()

p = Person(name="TWZ")

p.age = 23

p.save()

Person.objects.get_or_create(name="WZT", age=23)

这种方法是防止重复很好的方法,但是速度要相对慢些,返回一个元组,第一个为Person对象,第二个为True或False, 新建时返回的是True, 已经存在时返回False.

获取对象的方法

Person.objects.all()

Person.objects.all()[:10] 切片操作,获取10个人,不支持负索引,切片可以节约内存

Person.objects.get(name=name)


# get是用来获取一个对象的,如果需要获取满足条件的一些人,就要用到filter

Person.objects.filter(name="abc") # 等于Person.objects.filter(name__exact="abc") 名称严格等于 "abc" 的人

Person.objects.filter(name__iexact="abc") # 名称为 abc 但是不区分大小写,可以找到 ABC, Abc, aBC,这些都符合条件

Person.objects.filter(name__contains="abc") # 名称中包含 "abc"的人

Person.objects.filter(name__icontains="abc") #名称中包含 "abc",且abc不区分大小写

Person.objects.filter(name__regex="^abc") # 正则表达式查询

Person.objects.filter(name__iregex="^abc")# 正则表达式不区分大小写


# filter是找出满足条件的,当然也有排除符合某条件的

Person.objects.exclude(name__contains="WZ") # 排除包含 WZ 的Person对象

Person.objects.filter(name__contains="abc").exclude(age=23) # 找出名称含有abc, 但是排除年龄是23岁的

常用方法

可以参考博文:Django QuerySet API

1.得到模型中的所有记录

publisher_list = Publisher.objects.all()

2.保存模型的一个对象

publish.save()

3.模型数据的过滤

Publisher.objects.filter(name='Apress')

4.得到特定记录

Publisher.objects.get(name="Apress")

5.数据记录排序

Publisher.objects.all().order_by("name")

6.数据记录逆向排序

Publisher.objects.all().order_by("-name")

7.返回限制记录

Publisher.objects.order_by('name')[0]
Publisher.objects.order_by('name')[0:2]

8.快捷更新记录

Publisher.objects.filter(id=52).update(name='Apress Publishing')

9.删除记录

Publisher.objects.all().delete()

10.Foreign Key 反向得到记录

publisher.book_set.all()

  book_set 只是一个 QuerySet,所以它可以像QuerySet一样,能实现数据过滤和分切

publisher.book_set.filter(name__icontains='django')

  属性名称book_set是由模型名称的小写(如book)加_set组成的

11.访问多对多值(Many-to-Many Values)

book.authors.all()

book.authors.filter(first_name='Adrian')

反向查询

author.book_set.all()

参考文档:

Django models 官方教程: https://docs.djangoproject.com/en/dev/topics/db/models/

Fields相关官方文档:https://docs.djangoproject.com/en/dev/ref/models/fields/

添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注