Python Django ORM 的工作原理
在 Web 开发中,处理数据库是非常常见的需求,尤其是在构建动态应用程序时。Django 作为一个流行的 Python Web 框架,提供了一套强大的工具帮助开发者轻松管理数据库。Django 的 ORM(对象关系映射,Object-Relational Mapping)就是其中的重要组成部分,它使开发者可以通过 Python 代码与数据库进行交互,而无需编写 SQL 语句。对于新手来说,理解 Django ORM 的工作原理将极大地简化数据库操作。
本文将深入解释 Django ORM 的基本原理,并通过具体的代码示例帮助你更好地理解如何使用它。
什么是 ORM?
在正式介绍 Django ORM 之前,首先了解一下 ORM 的基本概念。ORM(Object-Relational Mapping,对象关系映射)是一种将面向对象编程语言中的对象与关系型数据库中的表进行映射的技术。简单来说,它允许你在代码中通过对象操作数据库,而不是通过 SQL 语句。
ORM 的工作机制是通过将数据库表映射到模型类(Model),每个类的实例代表数据库中的一条记录。通过 ORM,你可以像操作 Python 对象一样操作数据库中的数据,这大大减少了学习 SQL 和写复杂查询的需求。
Django ORM 是 Django 框架内置的 ORM 系统,它通过模型类定义数据库结构,并允许你通过 Python 代码与数据库进行交互。接下来,我们将详细介绍 Django ORM 的工作流程。
Django ORM 的基本工作原理
在 Django 中,ORM 的主要职责是将 Python 对象与数据库表相关联。这包括以下几个核心步骤:
- 定义模型类:每个模型类都映射到数据库中的一张表,类的属性对应于表中的字段。
- 迁移数据库结构:Django 提供了迁移(migrations)机制,它可以将模型类的结构变化反映到数据库中,自动生成 SQL 语句并执行。
- 操作数据库:通过 Django ORM,开发者可以通过 Python 代码执行 CRUD 操作(创建、读取、更新和删除),而无需直接编写 SQL 语句。
- 查询优化和链式查询:ORM 提供了高效的查询工具,可以通过链式调用构建复杂的查询逻辑,并保证执行效率。
1. 定义模型类
在 Django 中,每个模型类都代表数据库中的一张表。要定义一个模型类,我们需要在应用的 models.py
文件中创建一个继承自 models.Model
的类。
例如,假设我们正在开发一个博客应用,需要一个用于存储文章的表。可以定义如下的模型类:
from django.db import modelsclass Post(models.Model):title = models.CharField(max_length=200)content = models.TextField()created_at = models.DateTimeField(auto_now_add=True)updated_at = models.DateTimeField(auto_now=True)def __str__(self):return self.title
在这个示例中,我们定义了一个 Post
类,表示博客文章。这个类有以下几个字段:
title
: 文章标题,使用CharField
,表示一个字符串字段,max_length=200
限制了字符长度。content
: 文章内容,使用TextField
,用于存储大量文本。created_at
: 文章创建的时间,使用DateTimeField
,并且设置了auto_now_add=True
,表示自动填充当前时间。updated_at
: 文章更新时间,使用DateTimeField
,并设置auto_now=True
,每次文章更新时会自动修改这个字段。
每个字段类型(CharField
、TextField
、DateTimeField
等)都对应数据库中的数据类型,Django ORM 会根据这些定义生成相应的 SQL 语句。
2. 创建数据库迁移
模型类定义好之后,接下来需要将这些变化应用到数据库中。这是通过 Django 的迁移系统完成的。迁移是 ORM 用来跟踪和管理数据库表结构变化的工具,它可以自动生成 SQL 语句并应用到数据库。
执行以下命令来创建迁移文件:
python manage.py makemigrations
这个命令会扫描模型的变化,并生成相应的迁移文件。这些文件包含了创建或修改数据库表的 SQL 语句。接着,运行以下命令来应用迁移并将这些变化同步到数据库:
python manage.py migrate
当你运行 migrate
命令时,Django 会根据迁移文件生成 SQL 语句并执行,从而在数据库中创建或修改相应的表。
3. 操作数据库
当模型和数据库同步好之后,我们可以通过 Django ORM 来操作数据库了。Django ORM 提供了一些方便的方法来执行 CRUD(创建、读取、更新、删除)操作。
1. 创建数据
你可以通过模型类创建数据库记录。以下是创建一条新的博客文章的示例:
from blog.models import Post# 创建一篇新的文章
new_post = Post(title="我的第一篇文章", content="这是我的第一篇博客文章的内容。")
new_post.save() # 保存到数据库
在上面的代码中,我们首先实例化了 Post
类,传入文章的标题和内容。调用 save()
方法后,Django ORM 会生成一个 INSERT
语句,并将数据写入到数据库中。
2. 查询数据
Django ORM 提供了强大的查询功能,你可以使用 Post.objects
来查询 Post
表中的数据。
- 查询所有记录:
posts = Post.objects.all() # 返回所有文章对象的列表
- 按条件查询:
post = Post.objects.get(id=1) # 根据主键 id 查询单条记录
- 过滤查询:
recent_posts = Post.objects.filter(created_at__year=2024) # 查询 2024 年发布的文章
在这里,filter()
方法返回一个满足条件的查询集(QuerySet)。Django 的查询集支持链式调用,你可以通过多个条件组合成复杂的查询。
3. 更新数据
要更新一条记录,首先需要先查询出该记录,然后修改字段并调用 save()
方法。
post = Post.objects.get(id=1)
post.title = "更新后的标题"
post.save() # 保存更改
Django ORM 会生成一个 UPDATE
语句,将修改后的数据更新到数据库。
4. 删除数据
要删除一条记录,你可以使用 delete()
方法:
post = Post.objects.get(id=1)
post.delete() # 删除这条记录
delete()
方法会生成 DELETE
语句,并从数据库中删除该记录。
4. 查询优化与链式查询
Django ORM 的查询集是惰性执行的,这意味着查询操作不会立即访问数据库,只有当需要访问结果时才会执行 SQL 查询。这种惰性执行机制提高了性能,因为它允许 Django 构建复杂的查询链,而无需在每一步都访问数据库。
1. 链式查询
你可以通过链式调用将多个查询条件组合起来。以下是一个示例:
posts = Post.objects.filter(title__contains="Django").filter(created_at__year=2024)
在这个示例中,filter()
方法被调用了两次,最终的 SQL 查询将包含两个条件,查询标题中包含 “Django” 且在 2024 年创建的文章。
2. 查询优化
Django ORM 还提供了查询优化工具,例如 select_related()
和 prefetch_related()
,用于优化涉及外键和多对多关系的查询。
select_related()
:用于减少外键查询的数据库访问次数。它通过一个JOIN
操作一次性获取相关表的数据。
posts = Post.objects.select_related('author').all() # 假设 Post 表有一个外键指向 Author 表
prefetch_related()
:用于处理多对多关系的查询优化。它通过执行两次独立查询来减少数据库访问次数。
posts = Post.objects.prefetch_related('tags').all() # 假设 Post 表有一个多对多的 tags 关系
通过这些查询优化工具,你可以显著提高数据库查询的效率,尤其在处理大型项目或复杂的数据库结构时。
Django ORM 的高级功能
除了基本的 CRUD 操作,Django ORM 还提供了一些高级功能,帮助开发者更灵活地与数据库进行交互。
1. 模型继承
Django 支持模型继承,这意味着你可以创建一个基类模型,并让其他模型从它继承字段和方法。以下是一个示例:
class BaseModel(models.Model):created_at = models.DateTimeField(auto_now_add=True)updated_at = models.DateTimeField(auto_now=True)class Meta:abstract = True # 抽象基类,不会生成表class Post(BaseModel):title = models.CharField(max_length=200)content = models.TextField()
在这个示例中,BaseModel
是一个抽象基类,它定义了 created_at
和 updated_at
字段。Post
类继承了这个基类,并自动拥有这两个字段。
2. 自定义管理器
Django ORM 允许你创建自定义管理器(Manager)来扩展查询集的功能。例如,假设我们想创建一个自定义查询集,只返回最近发布的文章,可以这样实现:
class PostManager(models.Manager):def recent_posts(self):return self.filter(created_at__gte=datetime.now()-timedelta(days=30))class Post(models.Model):title = models.CharField(max_length=200)content = models.TextField()objects = PostManager() # 使用自定义管理器
现在我们可以调用 Post.objects.recent_posts()
来获取最近 30 天内发布的文章。
结论
Django ORM 提供了强大而灵活的工具,帮助开发者通过 Python 代码与数据库交互。它将面向对象编程的理念引入到数据库操作中,大大减少了编写 SQL 的需求,简化了 Web 开发中的数据库管理工作。通过本文的介绍,你应该能够理解 Django ORM 的基本原理,并开始在项目中使用它进行数据库操作。
相关文章:

Python Django ORM 的工作原理
在 Web 开发中,处理数据库是非常常见的需求,尤其是在构建动态应用程序时。Django 作为一个流行的 Python Web 框架,提供了一套强大的工具帮助开发者轻松管理数据库。Django 的 ORM(对象关系映射,Object-Relational Map…...

GoLang编程常用规范/工具
代码格式化工具 代码风格: go install github.com/fsgo/go_fmt/cmd/gorgeouslatest gorgeous # 仅格式化git有改动的文件。如需格式化所有文件,可以执行 gorgeous ./...标签格式:https://github.com/4meepo/tagalign go install github.c…...

数字王国里的虚拟人――技术、商业与法律解读
💂 个人网站:【 摸鱼游戏】【神级代码资源网站】【海拥导航】🤟 找工作,来万码优才:👉 #小程序://万码优才/HDQZJEQiCJb9cFi💅 想寻找共同学习交流,摸鱼划水的小伙伴,请点击【全栈技…...

Java后端基础练习|请求参数
请求参数,可以通过四种方式传递到后端 请求路径查询参数请求体请求头 controller代码 package com.urfread.breaknews.core.controller;import com.urfread.breaknews.core.common.model.ResultData; import lombok.Data; import org.springframework.web.bind.a…...

C# + SQLiteExpert 进行(cipher)加密数据库开发+Costura.Fody 清爽发布
一:让 SQLiteExpert 支持(cipher)加密数据库 SQLiteExpert 作为SQlite 的管理工具,默认不支持加密数据库的,使其成为支持(cipher)加密数据库的管理工具,需要添加e_sqlcipher.dll &…...

MySQL 8.0 新特性之自增变量持久化
MySQL 8.0 新特性之自增变量持久化 文章目录 MySQL 8.0 新特性之自增变量持久化MySQL 5.7 vs 8.0 测试对比MySQL 5.7MySQL 8.0 参考资料 MySQL 8.0 中支持自增变量持久化,实际也是解决之前版本中存在的自增主键重启重置的 BUG 问题( BUG #199࿱…...

网站建设公司哪家好?好的网站建设公司应该有哪些特别之处?
面对众多的网站建设公司,企业该如何选择呢?如何才能避坑呢?本文将探讨好的网站建设公司应该具备的特别之处 案例,这是最直观的表现,一个好的网站建设公司必然拥有为数众多的的案例展示,且这些案例质量高&a…...

香山南湖架构分析--FE
总体架构 分支预测和指令缓存,通过FTQ达到解耦的目的;FTQ将请求送给ICache,进行取指;取出的指令码通过预译码初步检查分支预测的错误并及时冲刷预测流水线;检查后的指令送入指令缓冲并传给译码模块,最终形成后端的指令…...

【Linux的那些事】shell命名及Linux权限的理解
目录 一、shell命令以及运行原理 二、Linux权限的概念 三、Linux权限管理 3.1.文件访问者的分类(人) 3.2.文件类型和访问权限(事物属性) 3.3.文件权限值的表示方法 3.4.文件访问权限的相关设置方法 a)chmod b)chown c)…...

Visual Studio 2022 配置 Boost 库
一、使用预编译版本 尽量不要使用预编译版本,因为可能构建的不完全,还得重新构建,不如一步到位 1. 下载预编译的 Boost 库 下载:Boost C Libraries - Browse /boost-binaries at SourceForge.net 2. 选择 msvc 版本࿰…...

逻辑回归(下): Sigmoid 函数的发展历史
背景 闲来无事翻了一下之前买的一个机器学习课程及之前记录的网络笔记,发现遇到公式都是截图,甚至是在纸上用笔推导的。重新整理一遍之前逻辑回归函数的学习笔记,主要是为了玩一下 LaTex 语法,写公式挺有意思的。 整理之前三篇笔…...

快速理解mQ(三)——RabbitMQ 各种交换机的区别与应用
RabbitMQ是一个开源的消息代理软件,它实现了高级消息队列协议(AMQP),允许应用程序或系统以异步的方式交换数据。RabbitMQ中的交换机(Exchange)是消息的分发中心,它接收来自生产者的消息…...

【五分钟学会】YOLO11 自定义数据集从训练到部署
数据集地址 数据集包含 360 张红血细胞图像及其注释文件,分为训练集与验证集。训练文件夹包含 300 张带有注释的图像。测试和验证文件夹都包含 60 张带有注释的图像。我们对原始数据集进行了一些修改以准备此 CBC 数据集,并将数据集分成三部分。在360张…...

Redis学习(十二)连接数不足报错及分析修复:ERR max number of clients reached.
目录 一、问题介绍二、问题分析2.1 redis-cli 登录2.2 info clients 查看连接数情况2.3 client list 查看具体连接情况2.4 分析连接空闲时长2.5 client list 根据客户端IP统计连接数 三、问题结论和解决3.1 问题结论:3.2 解决方案①:优化程序3.3 解决方案…...

八股文面试题总结(包含主流的面试经典题)
Java基础 1、JDK和JRE的区别是什么** JDK是Java开发工具包,JRE是Java运行时环境,二者的区别在于 JRE是Java程序运行所必须的,它包含jvm和一些Java的基础类库 JDK是Java程序开发所必须的,它包含JRE和一些开发工具 总结一下就是…...

一分钟掌握 Java22 新特性
1. 增强的模式匹配(Pattern Matching for Switch) Java 22 扩展了模式匹配的功能,允许在 switch 语句中使用模式匹配,从而使代码更加简洁和易读。 示例代码: public class PatternMatchingSwitchDemo {public stati…...

西安凭借入驻企业展示科技“硬”实力的数字媒体产业园
在古城西安的怀抱中,一座以科技“硬”实力为核心竞争力的数字媒体产业园——西安国际数字影像产业园,正以其独特的魅力和无限的潜力,吸引着全球的目光。这里,不仅是数字创意的孵化场,更是科技创新的策源地。 西安国际数…...

【网络安全】利用XSS、OAuth配置错误实现token窃取及账户接管 (ATO)
未经许可,不得转载。 文章目录 正文正文 目标:target.com 在子域sub1.target.com上,我发现了一个XSS漏洞。由于针对该子域的漏洞悬赏较低,我希望通过此漏洞将攻击升级至app.target.com,因为该子域的悬赏更高。 分析认证机制后,我发现: sub1.target.com:使用基于Cook…...

浙江所有省级医院体检报告查询上线浙里办!
在医院完成体检后不知道什么时候出报告 体检报告出来后又要跑一次医院去拿报告 历年体检报告没保管好 往年体检报告找不到了 ………… 编辑 为解决这些问题,浙江省卫生健康委结合“两卡融合、一网通办”工作的推进,不断丰富电子健康医保卡的功能&#…...

支付宝支付Java+uniapp支付宝小程序
JS: request.post(/vip/pay, {//这是自己写的java支付接口id: this.vipInfo.id,payWay: alipay-mini}).then((res) > {let success (res2) > {//前端的支付成功回调函数this.$refs.popup.close();// 支付成功刷新当前页面setTimeout(() > {this.doGetVipI…...

Linux-磁盘优化的几个思路
优化前先看看性能指标 I/O 基准测试 fio(Flexible I/O Tester)正是最常用的文件系统和磁盘 I/O 性能基准测试工具。这类工具非常用,了解即可 应用程序优化 应用程序处于整个 I/O 栈的最上端,它可以通过系统调用,来调…...

【第三版 系统集成项目管理工程师】第15章 组织保障
持续更新。。。。。。。。。。。。。。。 【第三版】第十五章 组织保障 15.1信息和文档管理15.1.1 信息和文档1.信息系统信息-P5462.信息系统文档-P546 15.1.2 信息(文档)管理规则和方法1.信息(文档)编制规范-P5472.信息(文档)定级保护-P5483.信息(文档)配置管理-P549练习 15.…...

从编程视角看生命、爱、自由、生活的排列顺序
从编程视角看生命、爱、自由、生活的排列顺序 离开大学校园,踏入大厂成为一名算法工程师后,我常以编程思维看待生活中的事物。在思考生命、爱、自由和生活的排列顺序时,发现从编程角度分析这些概念会有独特见解。 一、生命——程序的核心代…...

Lumerical——属性编辑窗口的详解
一、几何选项卡(Geometry tab) 通过几何选项卡中的选项可以设定物件的位置和大小。 二、材料选项卡(Material tab) ① 材料(MATERIAL): 在该字段可以设置成材料数据库中包含的任何材料。数据库也有可能包含一些新材料,也可以对已经包含的材料进行编辑。如果选…...

08实战篇:972应用题(2024)思路解析
在本节中,你将学习到: 2024年972应用题的基本解题思路如何应对陌生的社会/学术热点问题,避免初见杀题目一:阐述搜索引擎的工作基本原理 关键词:搜索引擎、基本原理题型:基础概念题这是一道混在应用题中的简答题。根据《信息检索》一书中的P37内容回答即可(同时对应P45—…...

解决应用程序启动失败问题:由于找不到d3dx9_43.dll文件,如何快速有效地恢复和修复缺失的DLL组件
由于找不到d3dx9_43.dll,许多用户在启动某些游戏或者图形相关软件时遇到了棘手的问题。这个小小的DLL文件,虽然看似不起眼,却在众多基于DirectX 9技术构建的应用程序中扮演着至关重要的角色。对于游戏爱好者来说,当满心欢喜地准备…...

Ubuntu——双系统Ubuntu22.04系统安装和基础配置
文章目录 一、Ubuntu22.04双系统安装双系统安装和卸载 二、基础配置更新软件源搜狗输入法TimeShift上网 三、AI配置CondaCUDA12.6Pytorch2.3.0-cu121 四、额外配置Samba局域网文件共享系统 一、Ubuntu22.04双系统安装 双系统安装和卸载 双系统卸载安装Ubuntu双系统windows系统…...

stm32定时器中断和外部中断
一,中断系统的介绍 中断:在主程序运行过程中,出现了特定的中断触发条件(中断源),使得CPU暂停当前正在运行的程序,转而去处理中断程序,处理完成后又返回原来被暂停的位置继续运行 中…...

LeetCode 每日一题 2024/9/30-2024/10/6
记录了初步解题思路 以及本地实现代码;并不一定为最优 也希望大家能一起探讨 一起进步 目录 9/30 1845. 座位预约管理系统10/1 983. 最低票价10/2 1870. 准时到达的列车最小时速10/3 1928. 规定时间内到达终点的最小花费10/4 1227. 飞机座位分配概率10/5 2187. 完成…...

Redis篇(最佳实践)(持续更新迭代)
介绍一:键值设计 一、优雅的key结构 Redis 的 Key 虽然可以自定义,但最好遵循下面的几个最佳实践约定: 遵循基本格式:[业务名称]:[数据名]:[id]长度不超过 44 字节不包含特殊字符 例如: 我们的登录业务࿰…...