ORM 相关操作
十三条常用方法
方法 | 解析 | 返回类型 |
---|---|---|
all() | 查询所有结果 | 返回QuerySet |
filter(**kwargs) | 它包含了与所给筛选条件相匹配的对象 | 返回QuerySet |
get(**kwargs) | 返回与所给筛选条件相匹配的对象,返回结果有且只有一个,如果符合筛选条件的对象超过一个或者没有都会抛出错误 | 返回具体对象 |
exclude(**kwargs) | 它包含了与所给筛选条件不匹配的对象 | 返回QuerySet |
values(*field) | 返回一个ValueQuerySet——一个特殊的QuerySet,运行后得到的并不是一系列model的实例化对象,而是一个可迭代的字典序列 | 特殊的QuerySet |
values_list(*field) | 它与values()非常相似,它返回的是一个元组序列,values返回的是一个字典序列 | 特殊的QuerySet |
order_by(*field) | 对查询结果排序 | 返回QuerySet |
reverse() | 对查询结果反向排序,请注意reverse()通常只能在具有已定义顺序的QuerySet上调用(在model类的Meta中指定ordering或调用order_by()方法) | 返回QuerySet |
distinct() | 从返回结果中剔除重复纪录(如果你查询跨越多个表,可能在计算QuerySet时得到重复的结果。此时可以使用distinct(),注意只有在PostgreSQL中支持按字段去重。) | 返回QuerySet |
count() | 返回数据库中匹配查询(QuerySet)的对象数量 | 返回数字 |
first() | 返回第一条记录 | 返回具体对象 |
last() | 返回最后一条记录 | 返回具体对象 |
exists() | 如果QuerySet包含数据,就返回True,否则返回False | 返回布尔值 |
神奇的双下方法
models.Tb1.objects.filter(id__lt=10, id__gt=1) # 获取id大于1 且 小于10的值
models.Tb1.objects.filter(id__in=[11, 22, 33]) # 获取id等于11、22、33的数据
models.Tb1.objects.exclude(id__in=[11, 22, 33]) # not in
models.Tb1.objects.filter(name__contains="ven") # 获取name字段包含"ven"的
models.Tb1.objects.filter(name__icontains="ven") # icontains大小写不敏感
models.Tb1.objects.filter(id__range=[1, 3]) # id范围是1到3的,等价于SQL的bettwen and
# 类似的还有:startswith,istartswith, endswith, iendswith
# date字段还可以:
models.Class.objects.filter(first_day__year=2017)
ForeignKey操作
表结构
class Book(models.Model):
"""这是一个图书类"""
name = models.CharField(max_length=50) # 书名
press = models.ForeignKey(to='Press')
def __str__(self):
return self.name
class Press(models.Model):
"""这是一个出版社类"""
name = models.CharField(max_length=50) # 出版社名称
def __str__(self):
return self.name
增
添加一个出版社
Press.objects.create(name='夜魅出版社')
Out[3]: <Press: 夜魅出版社>
或
Press(name='梦魇出版社').save()
添加一本书
Book.objects.create(name='斗罗大陆', press_id=1)
Out[5]: <Book: 斗罗大陆>
或
pre = Press.objects.create(name='火星出版社')
Book.objects.create(name='直到最后一句', press=pre)
Out[7]: <Book: 直到最后一句>
删
删除一本书
Book.objects.fileter(id=1).delete()
删除一个出版社
Press.objects.filter(id=3).delete()
Out[10]: (1, {'orm.Book': 0, 'orm.Press': 1})
改
book_obj = Book.objects.first()
book_obj.name='斗罗大陆Ⅱ'
book_obj.save()
或
Press.objects.filter(id=2).update(name='地球出版社')
注:
update 是 QuerySet的方法
查
正向查找
通过图书找出版社
Book.objects.first().press.name
# 第一本书对应的出版社的名字
Out[18]: '夜魅出版社'
拓展 通过图书修改出版社的名字
press_obj = Book.objects.first().press
# 获取第一本图书的出版社对象
press_obj.name='夜魅'
press_obj.save()
反向查找
通过出版社遭到图书
Press.objects.first().book_set.all()
# 获取第一个出版社 出版的所有书籍
Out[25]: <QuerySet [<Book: 斗罗大陆Ⅱ>]>
通过出版社修改出版社下书的名字
Press.objects.first().book_set.all().update(name='斗罗大陆Ⅲ')
Out[26]: 1
ManyToManyField
自动创建第三张表
表结构
class Teacher(models.Model):
"""这是一个老师类"""
name = models.CharField(max_length=50)
student = models.ManyToManyField(to='Student')
def __str__(self):
return self.name
class Student(models.Model):
"""这是一个学生类"""
name = models.CharField(max_length=50)
def __str__(self):
return self.name
增
正向添加
Teacher.objects.create(name='Teacher1').student.add(*Student.objects.all())
# 创建一个名字为Teacher1的教师,并在关联表中关联所有学生
或
Teacher.objects.create(name='T3').student.set([2,3,4])
反向添加
Student.objects.create(name='Student1').teacher_set.add(1)
# 创建一个名字为Student1的学生, 并关联教师id=1的教师
或
student_obj = Student.objects.create(name='S2')
student_obj.teacher_set.set(Teacher.objects.all())
删
删除是指删除第三张表中的数据
- clear() 清除所有关联
- remove() 删除对应关联
正向删除
Teacher.objects.first().student.clear()
# 清除教师表中第一个数据的所有关联
Teacher.objects.filter(id=2)[0].student.remove(*[1,2,3])
# 清除教师表中id为2 和学生id 为 1,2,3 的关联
反向删除
Student.objects.first().teacher_set.clear()
# 清除学生表中第一个数据的所有关联
Student.objects.filter(id=2)[0].teacher_set.remove(*[1,2,3])
# 清除学生表中id为2 和教师id 为 1,2,3 的关联
改
正向修改
Teacher.objects.first().student.set([1,3])
反向修改
Student.objects.first().teacher_set.set(Teacher.objects.all())
查
正向查找
Teacher.objects.filter(id=3)[0].student.all()
Teacher.objects.filter(id=3)[0].student.count()
Out[12]: 2
反向查找
Student.objects.first().teacher_set.all()
跨表查询
# 正向跨表
Teacher.objects.filter(id=3).values('id', 'name', 'student__name')
# 反向跨表
Student.objects.filter(id=1).values_list('id', 'teacher__name')
Q查询 与 F查询
F查询
Django 提供 F() 来做这样的比较。F() 的实例可以在查询中引用字段,来比较同一个 model 实例中两个不同字段的值。
示例1:
查询评论数大于收藏数的书籍
from django.db.models import F
models.Book.objects.filter(commnet_num__lt=F('keep_num'))
Django 支持 F() 对象之间以及 F() 对象和常数之间的加减乘除和取模的操作。
models.Book.objects.filter(commnet_num__lt=F('keep_num')*2)
修改操作也可以使用F函数,比如将每一本书的价格提高30元
models.Book.objects.all().update(price=F("price")+30)
引申:
如果要修改char字段咋办?
如:把所有书名后面加上(第一版)
>>> from django.db.models.functions import Concat
>>> from django.db.models import Value
>>> models.Book.objects.all().update(title=Concat(F("title"), Value("("), Value("第一版"), Value(")")))
Q查询
filter() 等方法中的关键字参数查询都是一起进行“AND” 的。 如果你需要执行更复杂的查询(例如OR语句),你可以使用Q对象。
示例1:
查询作者名是小仙女或小魔女的
from django.db.models import Q
models.Book.objects.filter(Q(authors__name="小仙女")|Q(authors__name="小魔女"))
你可以组合& 和| 操作符以及使用括号进行分组来编写任意复杂的Q 对象。同时,Q 对象可以使用~ 操作符取反,这允许组合正常的查询和取反(NOT) 查询。
示例:查询作者名字是小仙女并且不是2018年出版的书的书名。
>>> models.Book.objects.filter(Q(author__name="小仙女") & ~Q(publish_date__year=2018)).values_list("title")
<QuerySet [('番茄物语',)]>
查询函数可以混合使用Q 对象和关键字参数。所有提供给查询函数的参数(关键字参数或Q 对象)都将”AND”在一起。但是,如果出现Q 对象,它必须位于所有关键字参数的前面。
例如:查询出版年份是2017或2018,书名中带物语的所有书。
>>> models.Book.objects.filter(Q(publish_date__year=2018) | Q(publish_date__year=2017), title__icontains="物语")
<QuerySet [<Book: 番茄物语>, <Book: 香蕉物语>, <Book: 橘子物语>]>
事物
原子性、一致性、隔离性、永久性
try:
from django.db import transaction
with transaction.atomic():
new_publisher = models.Publisher.objects.create(name="火星出版社")
models.Book.objects.create(title="橘子物语", publish_date=datetime.date.today(), publisher_id=10) # 指定一个不存在的出版社id
except Exception as e:
print(str(e))
执行原生SQL语句
# extra
# 在QuerySet的基础上继续执行子语句
# extra(self, select=None, where=None, params=None, tables=None, order_by=None, select_params=None)
# select和select_params是一组,where和params是一组,tables用来设置from哪个表
# Entry.objects.extra(select={'new_id': "select col from sometable where othercol > %s"}, select_params=(1,))
# Entry.objects.extra(where=['headline=%s'], params=['Lennon'])
# Entry.objects.extra(where=["foo='a' OR bar = 'a'", "baz = 'a'"])
# Entry.objects.extra(select={'new_id': "select id from tb where id > %s"}, select_params=(1,), order_by=['-nid'])
# 举个例子:
models.UserInfo.objects.extra(
select={'newid':'select count(1) from app01_usertype where id>%s'},
select_params=[1,],
where = ['age>%s'],
params=[18,],
order_by=['-age'],
tables=['app01_usertype']
)
"""
select
app01_userinfo.id,
(select count(1) from app01_usertype where id>1) as newid
from app01_userinfo,app01_usertype
where
app01_userinfo.age > 18
order by
app01_userinfo.age desc
"""
# 执行原生SQL
# 更高灵活度的方式执行原生SQL语句
# from django.db import connection, connections
# cursor = connection.cursor() # cursor = connections['default'].cursor()
# cursor.execute("""SELECT * from auth_user where id = %s""", [1])
# row = cursor.fetchone()
Django终端打印SQL语句
在Django项目的settings.py文件中,在最后复制粘贴如下代码:
LOGGING = {
'version': 1,
'disable_existing_loggers': False,
'handlers': {
'console':{
'level':'DEBUG',
'class':'logging.StreamHandler',
},
},
'loggers': {
'django.db.backends': {
'handlers': ['console'],
'propagate': True,
'level':'DEBUG',
},
}
}
即为你的Django项目配置上一个名为django.db.backends的logger实例即可查看翻译后的SQL语句。