社区所有版块导航
Python
python开源   Django   Python   DjangoApp   pycharm  
DATA
docker   Elasticsearch  
aigc
aigc   chatgpt  
WEB开发
linux   MongoDB   Redis   DATABASE   NGINX   其他Web框架   web工具   zookeeper   tornado   NoSql   Bootstrap   js   peewee   Git   bottle   IE   MQ   Jquery  
机器学习
机器学习算法  
Python88.com
反馈   公告   社区推广  
产品
短视频  
印度
印度  
Py学习  »  Django

一篇文章带你了解Django ORM操作(基础篇)

Python程序员 • 3 年前 • 269 次点击  


前言

在日常开发中,需要大量数据库进行增删改查操作

如果头铁的话,使用原生SQL是最好的,毕竟性能又高,又灵活

但是通常情况下,我们不是太需要那么苛刻的性能,也没有那么多刁钻的需求原生SQL

通常会使用简单快捷的ORM进行增删改查

一起看学习一下Django的ORM操作吧

表结构设计

还是从实际角度出发。

假设,现在我需要设计一个简单的图书管理系统,是那种买的书,不是图书馆的书!!!

我想了想,首先,肯定有一个图书表,专门存放图书的信息,最起码是这样的。

但是又想了想,似乎我的书想发布,肯定是需要一个出版社帮我发布的,我肯定不能自己发布所以是这样子的。

又想了想,书肯定是人写的,肯定要有作者,所以还需要一个作者表,应该是这样子的。

又想了想,这一本书如果让一般人编,难度有点大啊,一般都是多人一起完成的,所以大概还有这样一张表。

其实,上述漏点了一个东西,图书需要归属一个出版社的,所以,最终表结构应该是这样的!

Django models代码

from django.db import models

# 作者表class Author(models.Model): name = models.CharField(verbose_name="作者姓名", max_length=8) age = models.IntegerField(verbose_name="作者年龄") phone = models.CharField(verbose_name="作者联系方式", max_length=11)

# 出版社class Publish(models.Model): title = models.CharField(verbose_name="出版社名称", max_length=16) phone = models.CharField(verbose_name="出版联系方式", max_length=11)

# 图书class Book(models.Model): title = models.CharField(verbose_name="书名", max_length=32) price = models.DecimalField(verbose_name="价格", max_digits=5, decimal_places=2) PublishDate = models.DateField(verbose_name="初版日期") publish = models.ForeignKey(to=Publish, verbose_name="所属出版社", on_delete=models.CASCADE)

# 图书Many作者class BookManyAuthor(models.Model): book = models.ForeignKey(to=Book, verbose_name="所属图书", on_delete=models.CASCADE) author = models.ForeignKey(to=Author, verbose_name="所属作者", on_delete=models.CASCADE)

Mysql.sql

数据同上述Excel图一致!

web_author.sql

web_book.sql

web_bookmanyauthor.sql

web_publish.sql

查询操作

本次采用单独使用Django ORM的方式,不需要将Django运行起来,所以也不需要写url什么的了!

前置导入

import osimport django
# django_orm_demo为我的项目名称os.environ.setdefault("DJANGO_SETTINGS_MODULE", "django_orm_demo.settings")django.setup()
# 导入models一定要在 django.setup() 之后from web import models

查询所有(all)

语法

models..objects.all()

例如:查询所有作者

author_list = models.Author.objects.all()print(author_list)

执行结果

例如:查询所有图书,并且遍历详情

book_list = models.Book.objects.all()for book in book_list:    print(book.title, book.price, book.PublishDate, book.publish)

执行结果

查询指定条件(filter)

语法

models..objects.filter()

例如:查询张三的信息

author = models.Author.objects.filter(name="张三")print(author)print(author.name)

执行结果

似乎你发现了个错误,'QuerySet' object has no attribute 'name'

这是因为filter()查询出来的,可能不止一个值,就像name=张三,可能有很多张三!

所以获取的是一个列表,注意第一个红色圈圈,如果我明明知道就是一个值,也就像取第一个值,咋办?

first

通过first,拿到的就是第一个值,同时也是模型类对象

代码

author = models.Author.objects.filter(name="张三").first()print(author,type(author))print(author.name,author.phone)

执行结果

filter支持多条件

假设有俩张三

如果我想取第二个张三咋办?

要是能写俩条件就好了比如这样name=张三 and age=22

注:filter后面只能跟filter或者first,filter里面的条件都是and查询

代码

# filter里面多个条件是and查询author = models.Author.objects.filter(name="张三",age=22).first()print(author,type(author))print(author.name,author.phone)

执行结果

常用filter条件列表

熟悉Mysql的可能都知道,有=,<,<=,like等各种范围查询,同样,Django也同样支持!

filter通过__来构造条件

# 包含三字段__contains="三" # 原生SQL条件:where 字段 like "%三%"# 以三开头字段__startswith="三"


    
 # 原生SQL条件:where 字段 like "三%"# 以三结尾字段__startswith="三" # 原生SQL条件:where 字段 like "%三"# 为空字段__isnull=True # 原生SQL条件:where 字段 IS NULL# 不为空字段__isnull=False # 原生SQL条件:where 字段 IS NOT NULL# in字段__in=[1,2,3] # 原生SQL条件:where 字段 IN (1, 2, 3)# >字段__gt=1 # 原生SQL条件:where 字段 > 1# >=字段__gte=1 # 原生SQL条件:where 字段 >= 1# <字段__lt=1 # 原生SQL条件:where 字段 < 1# <=字段__lte=1 # 原生SQL条件:where 字段 <= 1# 日期字段,年日期字段__year=2020 # 原生SQL条件:where 日期字段 BETWEEN 2020-01-01 AND 2020-12-31# 日期字段,月日期字段__month=3 # 原生SQL条件:where EXTRACT(MONTH FROM 表名.日期字段) = 3# 日期字段,天日期字段__day=4 # 原生SQL条件:where EXTRACT(DAY FROM 表名.日期字段) = 4# 时间比大小,|为或的意思日期字段__gt|lt|...="2020-10-10" # 原生SQL条件:where 日期字段 > 2020-10-10# 比大小还可以是时间类型import datetime日期字段__gt|lt|...=datetime.date(2020,10,10)# 同上# 时间范围筛选日期字段__range=("2020-01-01","2020-06-01") # 原生SQL条件:where 时间字段 BETWEEN 2020-01-01 AND 2020-06-01日期字段__range=(datetime.date(2020,1,1),datetime.date(2020,6,1)) # 同上

get

其实当我们只需要获取一个值时,还可以使用get

代码

author = models.Author.objects.get(name="李四")print(author,type(author))print(author.name,author.phone)

执行结果

但是这个get不太推荐使用,原因如下

  • 如果get条件获取了俩或俩以上的值,会报错。

  • 如果get条件获取不到值,还是会报错。

示例代码

author = models.Author.objects.get(name="张三")# error:get() returned more than one Author -- it returned 2!
author = models.Author.objects.get(name="不存在")# error:Author matching query does not exist.author = models.Author.objects.filter(name="不存在").first()# 结果:None

所以,推荐使用filter,如果确定只有一条,那就filter().first(),如果需要多条,遍历即可!

query

在某些特殊情况下,我们可能对于查询的结果感到有些意外!

可能需要看看原生SQL是啥,这时候需要用到query

注:query只能用在filter()后面。

代码

sql = models.Author.objects.filter(name="李四").queryprint(sql)

执行结果

总结

本篇通过一个类似实际的需求,进行了一个表结构设计。

书表,出版社表,作者表,图书和作者多对多表

以这几张表为例,进行Django ORM的学习。

学习了如何查询所有,如何条件查询,filter常用条件有哪些

get和filter().first()区别,如何通过query进行查看原生SQL

如果在操作过程中有任何问题,记得下面留言,我们看到会第一时间解决问题。

用微笑告诉别人,今天的我比昨天强,今后也一样。

如果你觉得文章还可以,记得点赞留言支持我们哈。感谢你的阅读,有问题请记得在下方留言噢~




福利

入门Python的最强三件套《ThinkPython》、《简明Python教程》、《Python进阶》的PDF电子版已打包提供给大家,关注下方公众号,在后台回复关键字P3」即可获取。

Python社区是高质量的Python/Django开发社区
本文地址:http://www.python88.com/topic/105928
 
269 次点击