Django笔记 —— 模型高级进阶
最近在学习Django,打算玩玩网页后台方面的东西,因为一直很好奇但却没怎么接触过。Django对我来说是一个全新的内容,思路想来也是全新的,或许并不能写得很明白,所以大家就凑合着看吧~
本篇笔记(其实我的所有笔记都是),并不会过于详细的讲解。因此如果有大家看不明白的地方,欢迎在我正版博客下留言,有时间的时候我很愿意来这里与大家探讨问题。(当然,不能是简简单单就可以百度到的问题-.-)
我所选用的教材是《The Django Book 2.0》,本节是第十章,模型高级进阶。
在基础部分的学习中,我体会到了一点经验:傻瓜教程最适合作为本书的笔记了~因为本书对于原理讲得很细,看一遍也就能基本理解,但由于讲得太细,具体操作步骤正是其不足。因此,读这本书,如果配上操作教程式的笔记,那复习起来就很舒服了 ^.^
因此,高级部分的笔记,将给出很多操作教程,没看过书的同学请先看了书再来看笔记~
0. 目录
1. 模型回顾与初探
1.1 访问外键(Foreign Key)值
1.2 访问多对多(Many-to-Many)值
2. 更改数据库模式(Database Schema)
3. Managers
3.1 增加额外Manager的方法
3.2 修改初始Manager QuerySets
4. 模型方法
5. 执行原始SQL查询
1. 模型回顾与初探
模型代码如下:
from django.db import modelsclass Publisher(models.Model):name = models.CharField(max_length=30)address = models.CharField(max_length=50)city = models.CharField(max_length=60)state_province = models.CharField(max_length=30)country = models.CharField(max_length=50)website = models.URLField()def __unicode__(self):return self.nameclass Author(models.Model):first_name = models.CharField(max_length=30)last_name = models.CharField(max_length=40)email = models.EmailField()def __unicode__(self):return u'%s %s' % (self.first_name, self.last_name)class Book(models.Model):title = models.CharField(max_length=100)authors = models.ManyToManyField(Author)publisher = models.ForeignKey(Publisher)publication_date = models.DateField()def __unicode__(self):return self.title
1.1 访问外键(Foreign Key)值
Book的外键是Publisher,首先,访问Book代码如下:
# Book>>> from mysite.books.models import Book >>> b = Book.objects.get(id=50) >>> b.title u'The Django Book'
然后,通过Book访问Publisher代码如下:
# Publisher>>> b = Book.objects.get(id=50) >>> b.publisher <Publisher: Apress Publishing> >>> b.publisher.website u'http://www.apress.com/'
通过Publisher访问Book的代码则如下:
# Book1>>> p = Publisher.objects.get(name='Apress Publishing') >>> p.book_set.all() [<Book: The Django Book>, <Book: Dive Into Python>, ...]# Book2>>> p = Publisher.objects.get(name='Apress Publishing') >>> p.book_set.filter(name__icontains='django') [<Book: The Django Book>, <Book: Pro Django>]
1.2 访问多对多(Many-to-Many)值
通过Book访问Author代码如下:
# Author>>> b = Book.objects.get(id=50) >>> b.authors.all() [<Author: Adrian Holovaty>, <Author: Jacob Kaplan-Moss>] >>> b.authors.filter(first_name='Adrian') [<Author: Adrian Holovaty>] >>> b.authors.filter(first_name='Adam') []
通过Author访问Book代码则如下:
# Book>>> a = Author.objects.get(first_name='Adrian', last_name='Holovaty') >>> a.book_set.all() [<Book: The Django Book>, <Book: Adrian's Other Book>]
2. 更改数据库模式(Database Schema)
当你修改模型时(例如添加字段、删除字段、删除模型……),应当依次进行以下步骤:
(1) 修改models.py中模型
(1+) 这时你的admin页面依旧正常运行,查看时也的确是修改完毕的样子,但是一旦添加(修改)对象则会报错
(2) 生成migration python manage.py makemigrations appname
(3) 激活模型(migrate) python manage.py migrate
(3+) 现在你可以正常使用了 ^.^
上述过程肯定是没问题的,除非你修改的是外键,那么就会很麻烦了。
例如,你删除了Book模型的外键,再要恢复则会在第2步遇到这样的提示:
You are trying to add a non-nullable field 'publisher' to book without a default; we can't do that (the database needs something to populate existing rows).
Please select a fix:
1) Provide a one-off default now (will be set on all existing rows)
2) Quit, and let me add a default in models.py
Select an option:
我不知道应该如何给值,于是选的2,结果……再也无法打开Book了……报错信息如下:
OperationalError at /admin/books/book/
(1054, "Unknown column 'books_book.publisher_id' in 'field list'")
因此,在我知道如何做之前,权宜之计,我不会尝试修改外键!
3. Managers
我们之前一直在使用的Book.objects.all()之类的语句,其Book.objects就是所谓的Manager,这是Django定义的用来管理模型的类,其中定义了很多函数,例如.all()。
而其中各种函数,例如all()、get(),返回的有类似列表的QuerySet,也有单个对象。
3.1 增加额外Manager的方法
有时候objects的功能不够用,我们就得写自己的Manager,代码如下:
1 # models.py 2 3 from django.db import models 4 5 # ... Author and Publisher models here ... 6 7 class BookManager(models.Manager): 8 def title_count(self, keyword): 9 return self.filter(title__icontains=keyword).count() 10 11 class Book(models.Model): 12 title = models.CharField(max_length=100) 13 authors = models.ManyToManyField(Author) 14 publisher = models.ForeignKey(Publisher) 15 publication_date = models.DateField() 16 num_pages = models.IntegerField(blank=True, null=True) 17 objects = BookManager() 18 19 def __unicode__(self): 20 return self.title
上述代码的第7~9行定义了自己的Manager,而第17行则把objects设为了这个Manager。
如此一来,我们便可以做如下操作了:
>>> Book.objects.title_count('django') 4 >>> Book.objects.title_count('python') 18
如果我们不定义objects,那么Django会自动生成;而现在咱们重新定义了objects,那么它就是咱们定义的这个Manager了。
多说一句,上面代码的作用,是查找所有书籍中名字带有指定字符串的书籍数量。
3.2 修改初始Manager QuerySets
我们在写出Book.objects()的时候,Django会认为我们选中了所有的Book对象,这是因为默认的objects类中有一个函数:get_query_set(),这个函数决定了你调用objects的时候选中哪些对象,它默认是返回模型中所有对象的。
那么,我们便可以改写一下这个函数,来让我们选中的对象(Query Set)改变一下,代码如下:
from django.db import models# First, define the Manager subclass. class DahlBookManager(models.Manager):def get_query_set(self):return super(DahlBookManager, self).get_query_set().filter(author='Roald Dahl')# Then hook it into the Book model explicitly. class Book(models.Model):title = models.CharField(max_length=100)author = models.CharField(max_length=50)# ... objects = models.Manager() # The default manager.dahl_objects = DahlBookManager() # The Dahl-specific manager.
如此一来,下面的Book.dahl_objects.all()便是上面函数所选的对象了:作者是Roald Dahol的书。
Book.dahl_objects.all() Book.dahl_objects.filter(title='Matilda') Book.dahl_objects.count()
再比如,我们可以在一个模型中实现几个不同的Manager,代码如下:
class MaleManager(models.Manager):def get_query_set(self):return super(MaleManager, self).get_query_set().filter(sex='M')class FemaleManager(models.Manager):def get_query_set(self):return super(FemaleManager, self).get_query_set().filter(sex='F')class Person(models.Model):first_name = models.CharField(max_length=50)last_name = models.CharField(max_length=50)sex = models.CharField(max_length=1, choices=(('M', 'Male'), ('F', 'Female')))people = models.Manager()men = MaleManager()women = FemaleManager()
4. 模型方法
我们之前一直在使用模型自带的方法,现在介绍如何自己定义方法:
from django.contrib.localflavor.us.models import USStateField from django.db import modelsclass Person(models.Model):first_name = models.CharField(max_length=50)last_name = models.CharField(max_length=50)birth_date = models.DateField()address = models.CharField(max_length=100)city = models.CharField(max_length=50)state = USStateField() # Yes, this is U.S.-centric...def baby_boomer_status(self):"Returns the person's baby-boomer status."import datetimeif datetime.date(1945, 8, 1) <= self.birth_date <= datetime.date(1964, 12, 31):return "Baby boomer"if self.birth_date < datetime.date(1945, 8, 1):return "Pre-boomer"return "Post-boomer"def is_midwestern(self):"Returns True if this person is from the Midwest."return self.state in ('IL', 'WI', 'MI', 'IN', 'OH', 'IA', 'MO')def _get_full_name(self):"Returns the person's full name."return u'%s %s' % (self.first_name, self.last_name)full_name = property(_get_full_name)
之后,使用代码如下:
>>> p = Person.objects.get(first_name='Barack', last_name='Obama') >>> p.birth_date datetime.date(1961, 8, 4) >>> p.baby_boomer_status() 'Baby boomer' >>> p.is_midwestern() True >>> p.full_name # Note this isn't a method -- it's treated as an attribute u'Barack Obama'
5. 执行原始SQL查询
最后,有时候会发现Django定义的内容不全,我们需要写一些sql代码,这时候就要:
>>> from django.db import connection >>> cursor = connection.cursor() >>> cursor.execute(""" ... SELECT DISTINCT first_name ... FROM people_person ... WHERE last_name = %s""", ['Lennon']) >>> row = cursor.fetchone() >>> print row ['John']
实际写入代码中的时候,最好写在自己的Manager中,就像这样:
from django.db import connection, modelsclass PersonManager(models.Manager):def first_names(self, last_name):cursor = connection.cursor()cursor.execute("""SELECT DISTINCT first_nameFROM people_personWHERE last_name = %s""", [last_name])return [row[0] for row in cursor.fetchone()]class Person(models.Model):first_name = models.CharField(max_length=50)last_name = models.CharField(max_length=50)objects = PersonManager()
如此,便可这样使用:
>>> Person.objects.first_names('Lennon') ['John', 'Cynthia']
至此,“模板高级进阶”内容完结,下一篇是——“通用视图”。
转载于:https://www.cnblogs.com/icedream61/p/4671736.html
Django笔记 —— 模型高级进阶相关推荐
- 碳中和丨InVEST模型高级进阶及在生态系统服务供需、碳中和等领域中的具体应用
InVEST 模型高级进阶及在生态 系统服务供需.碳中和等领域中的具体应用 1.原理深入浅出的讲解: 2.技巧方法讲解,提供所有案例数据及代码: 3.与项目案例相结合讲解实现方法,对接实际工作应用 : ...
- django1.8读书笔记模版高级进阶
一.概述 想要定制或者扩展模版引擎,模版系统工作原理,自动转移特征 名词解析:模板 渲染 就是是通过从context获取值来替换模板中变量并执行所有的模板标签. 二.Context处理器 如果在模版中 ...
- python框架django入门_Python高级进阶(一)Python框架之Django入门
姓名:{{name}} 爱好: {% for hobby in hobby_list %} {{hobby}}{% endfor %}
- 2019年最新面试动脑学院安卓 笔记安卓高级进阶-连接天天模拟器和adb常用命令
adb 重启 动脑学院安卓 企鹅找我 two2, zero0, eight8, one1 ,two2, eight8, four4, five5, eight8 ,one1 adb kill-ser ...
- Django基础(6): 模型Models高级进阶必读。
你或许已经早已读过我们的原创文章Django基础(1): 模型Models的介绍与设计,并已经知道一个模型的设计是一个app的核心.然而仅知道基础知识是远远不够的,在实际web开发过程中你需要掌握一些 ...
- keras提取模型中的某一层_Tensorflow笔记:高级封装——Keras
前言 之前在<Tensorflow笔记:高级封装--tf.Estimator>中介绍了Tensorflow的一种高级封装,本文介绍另一种高级封装Keras.Keras的特点就是两个字--简 ...
- python 科学计算三维可视化笔记(第三周 高级进阶)
python 科学计算三维可视化笔记 第三周 高级进阶 python 科学计算三维可视化笔记 第三周 高级进阶 一.Mayavi 入门 1. Mayavi 库的基本元素 2. 快速绘制实例 3. Ma ...
- Django框架——模型层单表操作、模型层多表操作、模型层常用和非常用字段和参数、模型层进阶
文章目录 1 模型层-单表操作 一 ORM简介 二 单表操作 2.1 创建表 1 创建模型 2 更多字段 3 更多参数 4 settings配置 5 增加,删除字段 2.2 添加表纪录 2.3 查询表 ...
- Flutter高级进阶实战 仿哔哩哔哩APP学习笔记
Flutter高级进阶实战 仿哔哩哔哩APP学习笔记 近年来各大公司对Flutter技能的要求越来越高,甚至设立了专门岗位,但掌握Flutter高阶技能的人才寥寥无几,市面上干货Flutter高阶课程 ...
- SWAT模型(建模方法、实例应用、高级进阶)技能提升
第一部分[从小白到精通]:SWAT建模及在水文水资源.面源污染模拟中的实践技术应用与典型案例分析 第二部分[高级进阶]:SWAT模型高阶应用暨无资料地区建模.不确定分析与气候变化.土地利用对面源污染影 ...
最新文章
- 生成静态文件的新闻系统核心代码
- java: web应用中不经意的内存泄露
- 为什么阿里巴巴禁止使用存储过程?
- Intellj idea启动tomcat配置
- 一个鸡蛋”改变TA的世界——让贫困地区的孩子每天都能吃上一个鸡蛋
- 使用delphi 开发多层应用(十)安全访问服务器
- fast路由器服务器未响应,win7系统下fast路由器进不了设置界面如何解决
- 面对锁等待难题,数仓如何实现问题的秒级定位和分析
- 基于matlab的车牌定位算法设计与实现,matlab车牌定位系统设计(源码+文档)
- 第一百七十三节,jQuery,Ajax
- linux重新开始学习
- 介绍一款界面自动化的工具
- linux 系统监控、诊断工具之 lsof 用法简介
- 程序阻碍OD调试的方式2
- 明解C语言 初级篇1~8章
- 使用h5的方式来实现钟表
- mysql配置kodi16.1_kodi中文包|kodi 16.1中文语言包下载 附插件使用教程 - 121下载站...
- OK资本周子涵:区块链的成果是“熬”出来的
- 原理竟然是这!mysql集群方案pxc
- 地方网站运营分享、未来发展汇报 [十年巨献,纯干货,不忽悠]
热门文章
- MySql 安装 Win python3
- BERT加速 | 预训练模型参数量越来越大?这里有你需要的BERT推理加速技术指南...
- 【论文详解】词向量ELMo: Embeddings from Language Models
- 利用python进行数据分析—四、Numpy基础:数组与向量化计算
- 李宏毅机器学习——结构化学习(一)
- 经典视觉SLAM框架
- 如何成为一名出色的演说者
- 5.12 CopyNet和 Pointer-Generator Net 复制机制和指针-生成器网络
- Android基本界面控件六-进度条
- mysql查看已打开文件数_[MySQL FAQ]系列 -- mysql如何计算打开文件数