回顾:

  • 定义 models

  • settings.py激活app才能使用models

  • migrations:版本控制,当更改库表结构时可以处理数据

  • 增删改查

  • 常见Field

模型的价值在于定义数据模型,使用Python的形式操作数据库

from django.db import models

class Person(models.Model):
first_name = models.CharField(max_length=30)
last_name = models.CharField(max_length=30)

转换为sql

CREATE TABLE IF NOT EXISTS myapp_person(
id INT(11) NOT NULL AUTO_INCREMENT,
first_name VARCHAR(30) DEFAULT NULL,
last_name VARCHAR(30) DEFAULT NULL,
PRIMARY KEY (id)
) ENGINE=InnoDB;

关系:

  • 一对多 ForeignKey,如厂商和汽车关系

  • 多对多 ManyToManyField,如用户和组关系

  • 一对一 OneToOneField,如饭店和地址关系

# 多对一关系
class Manufacturer(models.Model):
pass
class Car(models.Model):
manufacturer = models.ForeignKey('Manufacturer') # 多对多关系
class Group(models.Model):
pass
class User(models.Model):
groups = models.ManyToManyField('Group') # 一对一关系
class Place(models.Model):
name = models.CharField(max_length=50)
address = models.CharField(max_length=80)
class Restaurant(models.Model):
place = models.OneToOneField('Place', primary_key=True)
serves_hot_dogs = models.BooleanField(default=False)
serves_pizza = models.BooleanField(default=False)

重写save()方法:

如保存记录之前,写入日志操作,注意一定要记得调用超类的save()方法,否则不会保存到数据库

class Blog(models.Model):
name = models.CharField(max_length=100)
tagline = models.TextField() def save(self, *args, **kwargs):
print('do something')
super(Blog, self).save(*args, **kwargs)
print('do something else')

模型继承:

  • 抽象基类

只想使用父类来持有一些信息,不想在每个子模型中都敲一遍,这个类永远不会单独使用,使用abstract表明基类,数据库不会建该表

如auth模块 User继承(AbstractUser)

class CommonInfo(models.Model):
name = models.CharField(max_length=100)
age = models.PositiveIntegerField()
class Meta:
abstract = True
ordering = ['name'] class Student(CommonInfo):
home_group = models.CharField(max_length=5)
class Meta(CommonInfo.Meta):
db_table = 'student_info'

说明:

Meta 也可以继承,如果子类不实现Meta则默认会继承

  • 多表继承

继承一个存在的模型,每个模型都有自己的数据库表。每一个层级下的每个model都是一个真正意义上完整的model,

每个model都有专属的数据表,都可以查询和创建数据表,

继承关系在子model和它的每个父类之间都添加一个链接(通过自动创建的OneToOneField来实现)

class Place(models.Model):
name = models.CharField(max_length=50)
address = models.CharField(max_length=80) class Restaurant(Place):
serves_hot_dogs = models.BooleanField(default=False)
serves_pizza = models.BooleanField(default=False)
  • 代理继承

使用多表继承时,model的每个子类都会自动创建一张新数据表,通常情况下,这正是我们想要的操作,

这是因为子类需要一个空间来存储不包含在基类中的字段数据。但有时,可能只想更改model在Python层的行为实现,

比如:更改默认的manager,或是添加一个新方法

代理继承可以做到:为原始模型创建一个代理。可以创建、删除、更新代理model的实例,而且所有数据都可以像使用原始model一样保存

不同之处在于:可以在代理model中改变默认的排序设置和默认的manager,不会对原始model产生影响

设置 Meta类中proxy的值为True,就完成了对代理model的声明

class Person(models.Model):
first_name = models.CharField(max_length=30)
last_name = models.CharField(max_length=30) class MyPerson(Person):
class Meta:
proxy = True def do_something(self):
pass

说明:

MyPerson只是一个代理,数据库中不会建立该表,MyPerson类和它的父类Person操作同一个数据表,

特别的是,Person的任何实例也可以通过MyPerson访问,反之亦然

  • 多重继承

多重继承和多表继承没什么大的差别,会自动添加两个OneToOneField,

需要指出,两个基类的默认自增id要重新命名,否则添加OneToOneField会有问题,也不允许子类重写父类字段

class Article(models.Model):
article_id = models.AutoField(primary_key=True) class Book(models.Model):
book_id = models.AutoField(primary_key=True) class BookReview(Book, Article):
pass

创建记录时,父类的各个字段需要填写

b = BookReview.objects.create(headline='T' , body='body' , title=' title')

Model querys

回顾:

  • 简单增删改查

  • 返回结果 object QuerySets

  • exclude

  • 反向查询 b.entry_set.all()    可起别名,返回QuerySets

  • 复杂查询 Q

  • 跨关联关系查询,__

  • QuerySet链式过滤

  • F查询

  • 限制返回个数

查询对象比较

比较两个model实例,实质在后台,它会比较两个模型主键的值

所有查询api

https://docs.djangoproject.com/en/1.11/ref/models/querysets/

进阶:

1.  annotate添加注释属性 与聚合函数一起使用返回聚合值,如统计问题的个数,q就有了choice_num这个属性

2.  aggregate 聚合函数返回结果字典

3.  聚合类函数,或注释函数,在django.db.models模块中

  • class Avg(expression, output_field=FloatField, **extra)

  • class Count(expression, distinct=False,  **extra)

  • class Max(expression, output_field=None,  **extra)

  • class Min(expression, output_field=None,  **extra)

  • class Sum(expression, output_field=None,  **extra)

  • class StdDev(expression, sample=False, **extra) # 标准差

  • class Variance(expression, sample=False, **extra) # 方差

4.  distinct() 去重

5.  values 返回ValuesQuerySet(类字典),而不是模型实例对象

6.  vlaues_list与values类似,但返回元组而不是字典

7.  defer only

8.  using使用哪个数据库

9.  select_for_update 返回一个queryset,会锁定相关行直到事务结束

在支持的数据库上面产生一个 SELECT...FOR UPDATE

10.  raw 执行原始sql

11.  get_or_create,有就获取;没就新增,注意返回结果类型和参数defaults含义

try:
obj = Person.objects.get(first_name='John', last_name='Lennon')
except Person.DoesNotExist:
obj = Person.objects.get(first_name='John', last_name='Lennon', birthday=(1993, 7, 1))
# Equal
obj, created = Person.objects.get_or_create(
first_name = 'John',
last_name = 'Lennon',
defaults={'birthday': date(1993, 7, 1)})

12.  update_or_create

13.  bulk_create

14.  in_bulk

15.  latest、earliest、first、last

事务:

事务是绑定在view中实现的

from django.db import transaction

@transaction.non_atomic_requests
def my_view(request):
print('do_stuff') @transaction.atomic
def viewfunc(request):
print('do_stuff') '''
警告!!!
虽然这种事务模式的优势在于它的简单性,但在访问量增长到一定的时候会造成很大的性能损耗。
这是因为为每一个视图开启一个事物会有一些额外的开销。
另外,这种性能影响还取决于你的应用程序的查询模式以及你的数据库对锁的处理是否高效。
'''

自定义管理器:

  • 添加额外管理器方法

默认objects添加自定义方法,方法需写原生sql。

添加额外管理器方法是类增加“表级”功能的首选方式(如果要添加行级功能,比如只对某个模型的实例起作用,应使用模型方法,而不是管理器方法)

额外管理器方法可以返回你想要的任何数据,而不需要返回一个查询及集

例如,下面这个自定义管理器提供一个with_counts()方法,它返回所有OpinionPoll对象的列表,列表的每项都有一额外num_reponses属性,

该属性保存一个聚合查询的结果(注:对应的是SQL查询语句中的COUNT(*)生成的项)

# 添加额外管理器方法
class PollManager(models.Manager):
def with_counts(self):
from django.db import connection
cursor = connection.cursor()
cursor.execute("""
SELECT p.id, p.question, p.poll_date, COUNT(*)
FROM polls_opinionpoll p, polls_resopnse r
WHERE p.id = r.poll_id
GROUP BY p.id, p.question, p.poll_date
ORDER BY p.poll_date DESC""")
result_list = []
for row in cursor.fetchall():
p = self.model(id=row[0], question=row[1], poll_date=row[2])
p.num_responses = row[3]
result_list.append(p)
return result_list class OpinionPoll(models.Model):
question = models.CharField(max_length=200)
poll_date = models.DateField()
objects = PollManager() # 关键点*** class Response(models.Model):
poll = models.ForeignKey('OpinionPoll')
person_name = models.CharField(max_length=50)
response = models.TextField()
  • 添加自定义manager 

简单的说,就是在模型里添加属性=models.Manager(),或者它的自定义子类,从而实现对库表的操作

class AuthorManager(models.Manager):
def get_queryset(self):
return super(AuthorManager, self).get_queryset().filter(role='A') class EditorManager(models.Manager):
def get_queryset(self):
return super(EditorManager, self).get_queryset().filter(role='E') class Person(models.Model):
first_name = models.CharField(max_length=50)
last_name = models.CharField(max_length=50)
role = models.CharField(max_length=1, choices=(('A', 'Author'), ('E', 'Editor')))
people = models.Manager() # 等同于原始 objects
authors = AuthorManager()
editros = EditorManager() Person.objects.all()
Person.people.all()
Person.authors.all()
Person.editros.all()

Django Model 进阶的更多相关文章

  1. Django model进阶

    Django-model进阶   QuerySet 可切片 使用Python 的切片语法来限制查询集记录的数目 .它等同于SQL 的LIMIT 和OFFSET 子句. >>> Ent ...

  2. Django——model进阶(待完成)

    https://www.cnblogs.com/yuanchenqi/articles/7570003.html 一.QuerySet 1.可切片 使用Python 的切片语法来限制查询集记录的数目  ...

  3. Django框架学习-Model进阶用法

    Model进阶用法 回顾 访问外键 访问多对多关系 更改数据库结构 当处理数据库结构改变时,需要注意到几点: 增加字段 首先在开发环境中: 再到产品环境中: 删除字段 删除多对多字段 删除model ...

  4. Python之路【第十七篇】:Django【进阶篇 】

    Python之路[第十七篇]:Django[进阶篇 ]   Model 到目前为止,当我们的程序涉及到数据库相关操作时,我们一般都会这么搞: 创建数据库,设计表结构和字段 使用 MySQLdb 来连接 ...

  5. Python之路【第十七篇】:Django【进阶篇】

    Python之路[第十七篇]:Django[进阶篇 ]   Model 到目前为止,当我们的程序涉及到数据库相关操作时,我们一般都会这么搞: 创建数据库,设计表结构和字段 使用 MySQLdb 来连接 ...

  6. Python开发【第二十二篇】:Web框架之Django【进阶】

    Python开发[第二十二篇]:Web框架之Django[进阶]   猛击这里:http://www.cnblogs.com/wupeiqi/articles/5246483.html 博客园 首页 ...

  7. Python之路,Day15 - Django适当进阶篇

    Python之路,Day15 - Django适当进阶篇   本节内容 学员管理系统练习 Django ORM操作进阶 用户认证 Django练习小项目:学员管理系统设计开发 带着项目需求学习是最有趣 ...

  8. Django model select的各种用法详解

    <Django model update的各种用法介绍>文章介绍了Django model的各种update操作,这篇文章就是她的姊妹篇,详细介绍Django model select的用 ...

  9. Django 2.0 学习(07):Django 视图(进阶-续)

    接Django 2.0 学习(06):Django 视图(进阶),我们将聚焦在使用简单的表单进行处理和精简代码. 编写简单表单 我们将用下面的代码,来替换之前的detail模板("polls ...

随机推荐

  1. gentoo freemind 安装设置

    安装 freemind 之后,感觉菜单上面的字体比较模糊,通过设置 tools --> preference 中的 defaults --> default fonts 里面 的 defa ...

  2. Android APK反编译(一)

    apk是安卓工程打包的最终形式,将apk安装到手机或者模拟器上就可以使用APP.反编译apk则是将该安卓工程的源码.资源文件等内容破解出来进行分析. 一.APK反编译基本原理 1.APK分析 apk文 ...

  3. 转载:Opencv调整运行窗口图片的大小

    本文来自:http://blog.csdn.net/cumtml/article/details/52807961 Opencv在运算时显示图片问题 总结在opencv中,图片显示的问题.简要解决图片 ...

  4. tomcat 8 在线管理admin配置

    在tomcat8下,更加注重安全性.如果要使用在管理控制台部署应用,需要修改更多的配置. 在$tomcat_base$/webapps/manager/META-INF/context.xml中 添加 ...

  5. nginx的一次工作记录

    upstream fazhi_ui{ server ; } upstream fazhi_api{ server ; } server { listen ; server_name localhost ...

  6. spring @transactional 注解事务

    1.在spring配置文件中引入<tx:>命名空间 <beans xmlns="http://www.springframework.org/schema/beans&qu ...

  7. 【转】vMAN 和 PVID

    vMAN关的情况下,如果用户的包内带有VLAN TAG,则以用户的TAG为准,如果用户的包内不带VLAN TAG,就打上PVID:vMAN开的情况下,无论用户的包内是否带有VLAN TAG,都强制在外 ...

  8. rpm包安装的nginx热升级

    文章目录一.本地环境基本介绍二.yum升级命令说明三.升级好nginx后如何不中断业务切换3.1.nginx相关的信号说明3.2.在线热升级nginx可执行文件程序一.本地环境基本介绍本次测试环境,是 ...

  9. Codeforces Round #437 E. Buy Low Sell High

    题意:买卖股票,给你n个数,你可以选择买进或者卖出或者什么都不做,问你最后获得的最大收益是多少. Examples Input 910 5 4 7 9 12 6 2 10 Output 20 Inpu ...

  10. cdnbest区域里快速配置全部节点的缓存

    1.在cdn后台区域中自定义区域配置中添加下面代码,具体参数也可自行调整,代码解释在文档最下面有 <!--#start --> <config> <lang>zh_ ...