Django ORM相关操作

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语句。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值