前言
上一篇中,介绍了django的单表操作,本篇描述一下django的多表操作,包括条件过滤、联表操作等.
一、双下划綫条件过滤:
1 | # 获取个数 |
二、其他操作
1 | # 执行原生SQL |
三、联表操作
1.一对多
表结构如下:1
2
3
4
5
6
7class Book(models.Model):
name = models.CharField(max_length=32)
class Author(models.Model):
name = models.CharField(max_length=32)
books = models.ForeignKey(to="Book",to_field="id")
查询:1
2
3
4
5
6
7
8#方式一
a1=models.Author.objects.filter(id=1).first()
a1.books.name #以属性的方式关联,Author类的对象通过books属性关联至Book对象,再通过.name属性获取Book对象的name字段
#方式二
res=models.Author.objects.filter(id=1).values("books_id","books__name")
print(res)
#在本地表通过ForeignKey关联,会在本地表内默认生成一个以指定属性名+'_'+'to_field字段名'的字段,例如这里会生成一个books_id字段,代指book.id字段,因此books_id不需跨表查询,books__id需跨表查询。通过双下划线,可以跨表关联字段,例如book__name字段代指book.name
改:
1 | models.Author.objects.filter(id=1).update("books":[b1,b2..]) |
总结:
一对多操作,单对象,通过”.”属性方法方式跨表操作,queryset通过”__”跨表取字段
2.多对多
方式一:
自定义第三张表:
1 | class Book(models.Model): |
通过普通表进行反向间接查找
a1=models.Author.objects.filter(id=1).first()
obj = a1.relation_set.all()
#obj取出的是所有包含a1记录的relation类中的对象,queryset格式,使用类名+"_set"的格式时,类名字母需小写
obj1 = a1.relation_set.select_related('b').all()
#obj1取出的是所有包含a1记录通过relation类关联的Book类的对象,queryset格式
for i in obj:
print(i.b.name)
for i in obj:
print(i.name)
2 通过关系表relation进行查找
obj = models.Relation.objects.filter(a__=1).all()
for i in obj:
print(i.b.name)
通过中间关系表,相较于上面的反向查找更好理解。
方式二
只定义两张表,关系表由django自动创建维护
1 | #models.py |
总结:在没有ForeignKey或者ManyToManyField的表内反向查找时,需要使用使用类名+”_set”的格式来获取对象
one to one/maney to maney/foreign key各种混合关系查询
1 | class HostInfo(models.Model): |
1 | # 根据ip地址查host,hostinfo |