Django 模型层
基本操作
1、meta类属性汇总
属性名 | 用法 | 举例代码 |
abstract |
如果设置abstract=True则这个模式是一个抽象基类 | |
db_table |
定义model在数据库中的表名称,如果不定义,会生成默认’应用名_模型名‘ |
|
get_latest_by |
用于指定默认的排序字段,支持DateField\DateTimeField\IntegerField |
|
managed |
True或False,定义django命令行migrate工具是否管理此模型 | |
order_with_respect_to |
根据给定的字段对 model 排序。在关联关系中,它经常用在根据目标对象对源对象排序的场合。举 个例子,一个 Answer 只关联一个 Question 对象,而一个 question 对象却可以关联多 个 answer 对象。根据 question 对 answer 排序, |
|
ordering |
定义默认的排序列,如果在字段前加"-"表示降序排列,不加则是升序排列。加?随机排列 |
|
permissions |
在创建对象时,添加到权限表当中的附加权限信息。Django 自动为每个设置了 admin 的对象创建了添 加,删除和修改的权限。 |
|
default_permissions |
默认包含('add','change','delete') | |
proxy |
如果 proxy = True ,表示该 model 是其父类的代理 | |
unique_together |
用来设置的不重复的字段组合,必须唯一(将两个字段做联合唯一): |
|
index_together |
通过字段名集合创建索引,会根据选项生成合适的CREATE INDEX语句 |
|
verbose_name |
指定该对象的一个可读性更好的唯一名字(单数): |
|
verbose_name_plural |
指定该对象的一个可读性更好的唯一名字(复数):默认是verbose_name+"s" |
|
字段名称限制:
一个字段名不能包含连续的一个以上的下划线,因为那是Django查询语句的语法。
2、字段列表纵览表
字段名 |
参数 |
意义 |
AutoField |
|
一个能够根据可用ID自增的 IntegerField |
BooleanField |
|
一个真/假(true/false)字段 |
CharField |
(max_length) |
一个字符串字段,适用于中小长度的字符串。对于长段的文字,请使用 TextField |
CommaSeparatedIntegerField |
(max_length) |
一个用逗号分隔开的整数字段 |
DateField |
([auto_now], [auto_now_add]) |
日期字段 |
DateTimeField |
|
时间日期字段,接受跟 DateField 一样的额外选项 |
EmailField |
|
一个能检查值是否是有效的电子邮件地址的 CharField |
FileField |
(upload_to) |
一个文件上传字段 |
FilePathField |
(path,[match],[recursive]) |
一个拥有若干可选项的字段,选项被限定为文件系统中某个目录下的文件名 |
FloatField |
(max_digits,decimal_places) |
一个浮点数,对应Python中的 float 实例 |
ImageField |
(upload_to, [height_field] ,[width_field]) |
像 FileField 一样,只不过要验证上传的对象是一个有效的图片。 |
IntegerField |
|
一个整数。 |
IPAddressField |
|
一个IP地址,以字符串格式表示(例如: "24.124.1.30" )。 |
NullBooleanField |
|
就像一个 BooleanField ,但它支持 None /Null 。 |
PhoneNumberField |
|
它是一个 CharField ,并且会检查值是否是一个合法的美式电话格式 |
PositiveIntegerField |
|
和 IntegerField 类似,但必须是正值。 |
PositiveSmallIntegerField |
|
与 PositiveIntegerField 类似,但只允许小于一定值的值,最大值取决于数据库. |
SlugField |
|
嵌条 就是一段内容的简短标签,这段内容只能包含字母、数字、下 划线或连字符。通常用于URL中 |
SmallIntegerField |
|
和 IntegerField 类似,但是只允许在一个数据库相关的范围内的数值(通常是-32,768到 +32,767) |
TextField |
|
一个不限长度的文字字段 |
TimeField |
|
时分秒的时间显示。它接受的可指定参数与 DateField 和 DateTimeField 相同。 |
URLField |
|
用来存储URL的字段。 |
USStateField |
|
美国州名称缩写,两个字母。 |
XMLField |
(schema_path) |
它就是一个 TextField ,只不过要检查值是匹配指定schema的合法XML。 |
3、通用字段参数列表(所有的字段类型都可以使用下面的参数,所有的都是可选的。)
参数名 |
意义 |
null |
如果设置为 True 的话,Django将在数据库中存储空值为 NULL 。默认为 False 。 |
blank |
如果是 True ,该字段允许留空,默认为 False 。 |
choices |
一个包含双元素元组的可迭代的对象,用于给字段提供选项。 |
db_column |
当前字段在数据库中对应的列的名字。 |
db_index |
如果为 True ,Django会在创建表格(比如运行 manage.py syncdb )时对这一列创建数据库索引。 |
default |
字段的默认值 |
editable |
如果为 False ,这个字段在管理界面或表单里将不能编辑。默认为 True 。 |
help_text |
在管理界面表单对象里显示在字段下面的额外帮助文本。 |
primary_key |
如果为 True ,这个字段就会成为模型的主键。 |
radio_admin |
默认地,对于 ForeignKey 或者拥有 choices 设置的字段,Django管理界面会使用列表选择框(<select>)。如果 radio_admin 设置为 True 的话,Django就会使用单选按钮界面。 |
unique |
如果是 True ,这个字段的值在整个表中必须是唯一的。 |
unique_for_date |
把它的值设成一个 DataField 或者 DateTimeField 的字段的名称,可以确保字段在这个日期内不会出现重复值。 |
unique_for_month |
和 unique_for_date 类似,只是要求字段在指定字段的月份内唯一。 |
unique_for_year |
和 unique_for_date 及 unique_for_month 类似,只是时间范围变成了一年。 |
verbose_name |
除 ForeignKey 、 ManyToManyField 和 OneToOneField 之外的字段都接受一个详细名称作为第一个位置参数。 |
4、django字段查询谓词表
django两种过滤器
- filter:返回符合条件的数据集
- exclude:返回不符合条件的数据集
查询条件用:双下划线链接的字段名称和谓词来表达查询条件
谓词 | 含义 | 实例 | 等价SQL语句 |
exact | 精确等于 | Entry.objects.get(id__exact=14) | SELECT ... WHERE id = 14 |
iecact | 大小写不敏感的等于 | Entry.objects.get(headlin__exact='I like this') | SELECT....WHERE upper(headline) = 'I LIKE THIS' |
contains | 模糊匹配 | Entry.objects.get(headline__contains='Lennon') | SELECT ... WHERE headline LIKE '%Lennon%'; |
in | 在列表中查找 | Entry.objects.filter(id__in=[1, 3, 4]) | SELECT ... WHERE id IN (1, 3, 4); |
gt gte lt lte | 比较大小等于 | Entry.objects.filter(id__gt=4) | SELECT ... WHERE id > 4; |
startswith | 以...开始查找 | Entry.objects.filter(headline__startswith='Will') | SELECT ... WHERE headline LIKE 'Will%'; |
endswith | 以...结尾查找 | Entry.objects.filter(headline__endswith='cats') | SELECT ... WHERE headline LIKE '%cats'; |
range | 范围查找 | start_date = datetime.date(2005, 1, 1) end_date = datetime.date(2005, 3, 31) Entry.objects.filter(pub_date__range=(start_date, end_date)) |
SELECT ... WHERE pub_date BETWEEN '2005-01-01' and '2005-03-31'; |
year month day | 按年 月 日 查找 | Entry.objects.filter(pub_date__year=2005) | SELECT ... WHERE pub_date BETWEEN '2005-01-01' AND '2005-12-31 23:59:59.999999'; |
week_day | 按星期几查找 | Entry.objects.filter(pub_date__week_day=2) | |
isnull | 是否为空查找 | Entry.objects.filter(pub_date__isnull=True) | SELECT ... WHERE pub_date IS NULL; |
Django还提供了get()用于查询单条记录 comment.objects.get(id__exact = 1)
Django还支持查询指定条数的数据集
comment.objects.all()[:10]
comment.objects.all()[10:20]
comment.objects.all()[1]
还支持order_by操作
comment.objects.order_by('headline')
5、数据保存与删除
DJango定义了一个统一的save方法,完成模型的insert和update操作,执行save函数,自动判断记录是否存在,存在则更新,否则删除
删除操作 comment.objects.get(id__exact = 1)。delete()
6、关系操作
一对一关系 OneToOneField
from django.db import models class Account(models.Model): username = models.CharField(max_length=80) password = models.CharField(max_length=225) ref_date = models.DateField() def __unicode__(self): return 'Account: %s' % self.username class Contact(models.Model): account = models.OneToOneField( Account, # 被关联的模型名 on_delete=models.CASCADE,# 当被关联模型记录被删除时,本记录也会被删除 primary_key=True, ) zip_code = models.CharField(max_length=10) address = models.CharField(max_length=80) mobile = models.CharField(max_length=20) def __unicode__(self): # 定义模型的显示字符串 return '%s ,%s' % (self.account.username,self.mobile)
一对多关系 ForeignKey
from django.db import models class Account(models.Model): username = models.CharField(max_length=80) password = models.CharField(max_length=225) ref_date = models.DateField() def __unicode__(self): return 'Account: %s' % self.username class Contact(models.Model): account = models.ForeignKey( Account, # 被关联的模型名 on_delete=models.CASCADE,# 当被关联模型记录被删除时,本记录也会被删除 ) zip_code = models.CharField(max_length=10) address = models.CharField(max_length=80) mobile = models.CharField(max_length=20) def __unicode__(self): # 定义模型的显示字符串 return '%s ,%s' % (self.account.username,self.mobile)
print(a.contact_set) # 从主对象中找到附对象 [<Contact:Rose,13145671234>,<Contact:Rose,13178654234>]
在一对多关系中,每个模型可以关联多个子对象,所以本例从主模型Account对象中寻找附模型Contact的属性时contact_set,通过一个集合返回关联结果
注:xxx_set时Django设定的通过主模型对象访问附模型对象集合的属性名,相当于flask中的backref属性
多对多关系 ManyToMany
from django.db import models class Account(models.Model): username = models.CharField(max_length=80) password = models.CharField(max_length=225) ref_date = models.DateField() def __unicode__(self): return 'Account: %s' % self.username class Contact(models.Model): accounts = models.ManyToManyField( Account, # 被关联的模型名 ) zip_code = models.CharField(max_length=10) address = models.CharField(max_length=80) mobile = models.CharField(max_length=20) def __unicode__(self): # 定义模型的显示字符串 return '%s ,%s' % (self.accounts.username,self.mobile)
演示
#分别建立和保存Account和Contact对象 >>a1 = Account(user_name = 'Leon') >>a1.save() >>c1 = Contact(mobile = '13145671234') >>c1.save() >>c1.accounts.add(a1) # 通过Contact对象建立关系 >>a2= Account(user_name = 'Terry') >>a2.save() >>a2.contact_set.add(c1) # 通过Account对象建立关系 >>a1.contact_set.remove(c1) # 取消单个对象关联 >>a1.contact_set.clear() # 取消a1与所有其他Contact对象的关联
7、面向对象ORM
抽象类继承
抽象类的作用是在多个表由若干相同的字段时,可以将这些字段统一定义在抽象基类中,免于重复定义这些字段,抽象基类通过模型的meta中定义abstract = True来实现
from django.db import models # Create your models here. class MessageBase(models.Model): id = models.AutoField() username = models.CharField(max_length=80) content = models.CharField(max_length=225) pub_date = models.DateField() class Meta: abstract=True # 定义本类为抽象基类 KIND_CHOICES = ( ('python技术','python技术'), ('数据库技术','数据库技术'), ('经济学','经济学'), ('文体资讯','文体资讯'), ('个人心情','个人心情'), ('其他','其他'), ) class Moment(MessageBase): #继承抽象类 kind = models.CharField(max_length=20,choices=KIND_CHOICES,default=KIND_CHOICES[0]) class Comment(MessageBase): headline = models.CharField(max_length=50)
子类模型的编程中可以直接引用父类定义的字段,以上会生成2个表
多表继承-无需特殊关键字
from django.db import models # Create your models here. class MessageBase(models.Model): id = models.AutoField() username = models.CharField(max_length=80) content = models.CharField(max_length=225) pub_date = models.DateField() KIND_CHOICES = ( ('python技术','python技术'), ('数据库技术','数据库技术'), ('经济学','经济学'), ('文体资讯','文体资讯'), ('个人心情','个人心情'), ('其他','其他'), ) class Moment(MessageBase): kind = models.CharField(max_length=20,choices=KIND_CHOICES,default=KIND_CHOICES[0]) class Comment(MessageBase): headline = models.CharField(max_length=50)
以上会生成3个表,但是编程中,django内部自动实现父模型和子模型之间一对一关系来实现多表继承技术
# 新建Comment对象,直接在子类中引用父类字段 >>m1 = Comment(username = 'join',headline = 'hello') >>m1.content = 'hahahahah' >>m1.save() # 通过父类实例引用父类字段 >> print(m1.messagebase.content)# 多表继承中通过小写的父类名字就可以引用父类实例 hahahahah
代理模型继承
前两种继承模型中子类模型都有实际存储的作用,而代理模型继承中子类只用于管理父类数据,而不实际存储数据,代理模型继承需要通过子类的Meta中定义的proxy = True属性来实现
from django.db import models class Moment(models.Model): id = models.AutoField() username = models.CharField(max_length=80) content = models.CharField(max_length=225) pub_date = models.DateField() class OrderdMoment(Moment): class Meta: proxy = True ordering =['-pub_date']
使用代理继承的原因时:子类中新的特性不影响父类模型及其已有代码的行为