Python 如何使用 SQLAlchemy 进行复杂查询
Python 如何使用 SQLAlchemy 进行复杂查询
一、引言
SQLAlchemy 是 Python 生态系统中非常流行的数据库处理库,它提供了一种高效、简洁的方式与数据库进行交互。SQLAlchemy 是一个功能强大的数据库工具,支持结构化查询语言(SQL)的映射,允许开发人员通过 Python 代码编写复杂的数据库查询操作,而无需直接编写原始 SQL 语句。
在数据驱动的应用程序中,复杂查询是必不可少的。为了从数据库中提取所需的信息,我们经常需要使用 JOIN、GROUP BY、ORDER BY、子查询等操作。SQLAlchemy 不仅支持这些复杂的查询,还提供了 ORM(对象关系映射)和核心层的 SQL 表达式语言,使我们可以以一种灵活和优雅的方式构建复杂的数据库查询。
本文将通过一些常见的示例介绍如何使用 SQLAlchemy 编写复杂查询。对于刚开始接触 SQLAlchemy 的新手来说,本文将会以通俗易懂的方式展示 SQLAlchemy 的查询能力,并结合实例代码帮助你更好地理解。

二、SQLAlchemy 简介
SQLAlchemy 提供了两个核心组件:
- ORM(对象关系映射):通过 Python 类映射到数据库表,实现以面向对象的方式与数据库交互。
- SQL 表达式语言:允许开发者使用 Python 表达式构建 SQL 查询,提供了更多低级别的 SQL 操作控制。
SQLAlchemy 的这两个组件可以单独使用,也可以结合使用。本文主要聚焦于 ORM 模式下如何使用 SQLAlchemy 进行复杂查询。
2.1 SQLAlchemy 安装
在使用 SQLAlchemy 之前,你需要确保已经安装了该库。可以通过 pip 命令安装:
pip install sqlalchemy
此外,如果你打算连接到 MySQL、PostgreSQL、SQLite 等数据库,还需要安装对应的数据库驱动程序。以下是安装常见数据库驱动的命令:
# 安装 MySQL 驱动
pip install pymysql# 安装 PostgreSQL 驱动
pip install psycopg2# SQLite 通常自带,无需额外安装
2.2 连接到数据库
在编写复杂查询之前,我们需要先连接到数据库并创建一个会话对象。SQLAlchemy 使用引擎(engine)对象来与数据库建立连接,并通过会话(session)对象管理事务和查询。
from sqlalchemy import create_engine
from sqlalchemy.orm import sessionmaker# 创建数据库引擎(以 SQLite 为例)
engine = create_engine('sqlite:///example.db')# 创建会话类
Session = sessionmaker(bind=engine)# 创建会话实例
session = Session()
在上面的代码中,我们创建了一个连接到 SQLite 数据库的引擎,并通过 sessionmaker 函数生成了会话类,最后创建了一个会话实例,用于后续的数据库操作。
三、定义模型(Model)
在使用 SQLAlchemy ORM 进行查询之前,首先需要定义数据库的表结构。在 SQLAlchemy 中,表结构通过 Python 类来定义,并通过类属性与数据库字段建立映射关系。
假设我们有一个简单的数据库,包含三个表:User、Post 和 Comment,它们分别表示用户、帖子和评论。我们将使用这些表来展示如何进行复杂查询。
from sqlalchemy import Column, Integer, String, ForeignKey
from sqlalchemy.orm import relationship
from sqlalchemy.ext.declarative import declarative_base# 创建模型基类
Base = declarative_base()# 定义 User 表
class User(Base):__tablename__ = 'users'id = Column(Integer, primary_key=True)name = Column(String)# 与 Post 关联posts = relationship("Post", back_populates="user")# 定义 Post 表
class Post(Base):__tablename__ = 'posts'id = Column(Integer, primary_key=True)title = Column(String)content = Column(String)user_id = Column(Integer, ForeignKey('users.id'))# 与 User 关联user = relationship("User", back_populates="posts")# 与 Comment 关联comments = relationship("Comment", back_populates="post")# 定义 Comment 表
class Comment(Base):__tablename__ = 'comments'id = Column(Integer, primary_key=True)content = Column(String)post_id = Column(Integer, ForeignKey('posts.id'))# 与 Post 关联post = relationship("Post", back_populates="comments")
在上面的代码中,我们定义了三个模型类:User、Post 和 Comment,它们分别映射到数据库中的三个表。我们使用 relationship() 方法建立了模型之间的关系,User 和 Post 是一对多的关系,而 Post 和 Comment 也是一对多的关系。
四、SQLAlchemy 中的复杂查询
接下来,我们将展示如何使用 SQLAlchemy 进行复杂的查询操作。
4.1 基本查询
最基本的查询是从一个表中检索所有的记录。SQLAlchemy 提供了 query() 方法用于执行查询操作。
# 查询所有用户
users = session.query(User).all()for user in users:print(user.name)
4.2 条件查询(WHERE)
在 SQLAlchemy 中,使用 filter() 方法可以为查询添加条件,类似于 SQL 中的 WHERE 子句。
# 查询名字为 'Alice' 的用户
alice = session.query(User).filter(User.name == 'Alice').first()
print(alice.name)
4.3 排序(ORDER BY)
可以通过 order_by() 方法对查询结果进行排序。
# 查询帖子并按照创建顺序排序
posts = session.query(Post).order_by(Post.id).all()for post in posts:print(post.title)
4.4 连接查询(JOIN)
连接查询(JOIN)是数据库查询中非常常见的操作,通常用于从多个表中获取数据。SQLAlchemy 通过 join() 方法支持连接查询。
# 查询每个帖子及其对应的用户信息
posts_with_users = session.query(Post, User).join(User).all()for post, user in posts_with_users:print(f"帖子标题: {post.title}, 作者: {user.name}")
4.5 分组查询(GROUP BY)
分组查询通常用于数据统计。SQLAlchemy 通过 group_by() 方法支持分组操作。
from sqlalchemy import func# 查询每个用户的帖子数量
user_post_count = session.query(User.name, func.count(Post.id)).join(Post).group_by(User.id).all()for name, count in user_post_count:print(f"用户: {name}, 帖子数量: {count}")
4.6 子查询
在某些情况下,我们需要在一个查询中嵌套另一个查询,即使用子查询。SQLAlchemy 提供了灵活的方式来构建子查询。
# 查询评论数量大于 2 的帖子
subquery = session.query(Comment.post_id, func.count(Comment.id).label('comment_count')).group_by(Comment.post_id).subquery()posts_with_many_comments = session.query(Post).join(subquery, Post.id == subquery.c.post_id).filter(subquery.c.comment_count > 2).all()for post in posts_with_many_comments:print(post.title)
4.7 复杂条件(AND、OR)
SQLAlchemy 支持通过 and_() 和 or_() 方法来构建复杂的查询条件。
from sqlalchemy import or_, and_# 查询名字为 'Alice' 或者帖子标题包含 'Python' 的帖子
results = session.query(Post).filter(or_(Post.user.has(User.name == 'Alice'),Post.title.like('%Python%'))
).all()for post in results:print(post.title)
4.8 分页查询
当数据量较大时,分页查询有助于提高性能。SQLAlchemy 支持通过 limit() 和 offset() 方法进行分页操作。
# 查询前 5 个帖子
first_five_posts = session.query(Post).limit(5).all()for post in first_five_posts:print(post.title)
五、SQLAlchemy 的优缺点
5.1 优点
- 简洁易用:SQLAlchemy 提供了简洁的 API,使我们能够通过 Python 代码轻松进行复杂的数据库操作。
- ORM 支持:SQLAlchemy 的 ORM 功能允许我们将数据库表映射为 Python 类,使得操作数据库如同操作普通对象。
- 灵活性:SQLAlchemy 同时支持高层次的 ORM 查询和底层的 SQL 表达式语言,使我们能够根据需求选择合适的查询方式。
- 数据库无关性:SQLAlchemy 可以支持多种数据库,包括 MySQL、PostgreSQL、SQLite 等。
5.2 缺点
- 学习曲线较陡:尽管 SQLAlchemy 的基本用法比较简单,但其高级功能
,如复杂查询和关系管理,可能需要更多的学习和实践。
2. 性能开销:在处理非常大的数据集时,使用 ORM 可能会带来一定的性能开销。
六、总结
通过本文的介绍,你应该对如何使用 SQLAlchemy 进行复杂查询有了更深入的了解。SQLAlchemy 提供了强大的 ORM 功能,使我们能够用面向对象的方式处理数据库操作。此外,SQLAlchemy 的 SQL 表达式语言也为我们提供了构建复杂查询的灵活性。
无论是简单的查询还是复杂的 JOIN、GROUP BY 和子查询,SQLAlchemy 都能够帮助我们高效地从数据库中提取数据。在实际开发中,选择合适的查询方式能够提高应用程序的性能,并减少代码的复杂性。
相关文章:
Python 如何使用 SQLAlchemy 进行复杂查询
Python 如何使用 SQLAlchemy 进行复杂查询 一、引言 SQLAlchemy 是 Python 生态系统中非常流行的数据库处理库,它提供了一种高效、简洁的方式与数据库进行交互。SQLAlchemy 是一个功能强大的数据库工具,支持结构化查询语言(SQL)…...
nginx主配置文件
Nginx的主配置文件nginx.conf,一般定义了Nginx的基本设置和全局配置。下面是对这个配置文件的详细解释: 文件结构 #user nobody; worker_processes 1;#error_log logs/error.log; #error_log logs/error.log notice; #error_log logs/error.log …...
使用数据库:
数据库: 1.为何需要数据库? 存储数据方法 第一种:用大脑记住数据, 第二种:写纸上, 第三种:写在计算机的内存中, 第四种:写出磁盘文件 2.数据库能做什么࿱…...
python list, tuple dict,set的区别 以及**kwargs 的基本用法
在python中, list, tuple, dict, set有什么区别, 主要应用在什么样的场景? 定义: list:链表,有序的项目, 通过索引进行查找,使用方括号”[]”; tuple:元组,元组将多样的对象集合到一起,不能修改,通过索引进行查找, 使用括号”()”; dict:字典,字典是一组键(key)和值(value…...
实用生活英语口语学习成人零基础入门柯桥专业外语培训
“秋裤”的英语表达 首先,秋裤肯定不是autumn pants,chill cool就更离谱了! 最地道的美语说法一定会用到“thermal”这个单词: ▼ “thermal”的意思是“热的、保温的”,由此延伸出“秋裤、保暖内衣”的表达ÿ…...
FLINK SQL数据类型
Flink SQL支持非常完善的数据类型,以满足不同的数据处理需求。以下是对Flink SQL数据类型的详细归纳: 一、原子数据类型 字符串类型 CHAR、CHAR(n):定长字符串,n代表字符的定长,取值范围为[1, 2147483647]。如果不指…...
汇编语言教程:打造你的第一款汇编语言小游戏 汇编语言教程攻略
目录 游戏详细简介 完整代码示例(不少于70行) 如何自学汇编语言游戏开发攻略及功能 游戏详细简介 游戏名称:“太空探险” 游戏简介:这是一款基于x86汇编语言开发的简单2D游戏。在游戏中,玩家扮演一名宇航员&#…...
白色简洁大方公司企业网站源码 WordPress主题2款
WordPress白色简洁大方公司企业网站主题2款 白色整洁风格wordpress主题是一款比较新颖的国际设计范风格 简洁而大方的 WordPress 主题,适合个人博客、企业和工作室用。 完美支持下拉菜单的wordpress企业主题。 wordpress简白企业模板是一款适合企业站以及工作室…...
MinIO分片上传超大文件(纯服务端)
目录 一、MinIO快速搭建1.1、拉取docker镜像1.2、启动docker容器 二、分片上传大文件到MinIO2.1、添加依赖2.2、实现MinioClient2.3、实现分片上传2.3.0、初始化MinioClient2.3.1、准备分片上传2.3.2、分片并上传2.3.2.1、设置分片大小2.3.2.2、分片 2.3.3、分片合并 三、测试3…...
leetcode链表(一)-移除链表元素
题目 t. - 力扣(LeetCode) 给你一个链表的头节点 head 和一个整数 val ,请你删除链表中所有满足 Node.val val 的节点,并返回 新的头节点 。 例1 输入:head [1,2,6,3,4,5,6], val 6 输出:[1,2,3,4,5]…...
python的特殊方法——魔术方法
前言 __init__(self[]) 编辑 __call__(self [, ...]) __getitem__(self, key) __len__(self) __repr__(self) / __str__(self) __add__(self, other) __radd__(self, other) 参考文献 前言 官方定义好的,以两个下划线开头且以两个下划线结尾来命名的方法…...
深入浅出理解TCP三次握手与四次挥手
目录 引言1.为什么需要三次握手?2. 三次握手的过程3. 为什么需要四次挥手?4. 四次挥手的过程5. 为什么挥手需要四次,而握手只需三次?6. 三次握手与四次挥手的时序图7. TIME_WAIT状态的意义8. 总结9.面试时候问到什么是三次握手和四…...
如何在Windows和Linux查看正在监听的端口和绑定的进程
端口(Port)和进程(Process)是计算机网络和操作系统中的重要概念,它们之间有着密切的关系。以下是对这两个概念的详细介绍以及它们之间的关系(附Windows和Linux查看端口和进程的命令): 端口(Por…...
如何用深度神经网络预测潜在消费者
1. 模型架构 本项目采用的是DeepFM模型,其结构结合了FM(因子分解机)与深度神经网络(DNN),实现了低阶与高阶特征交互的有效建模。模型分为以下几层: 1.1 FM部分(因子分解机层&#…...
基于opencv答题卡识别判卷
我们是一个深度学习领域的独立工作室。团队成员有:中科大硕士、纽约大学硕士、浙江大学硕士、华东理工博士等,曾在腾讯、百度、德勤等担任算法工程师/产品经理。全网20多万粉丝,拥有2篇国家级人工智能发明专利。 社区特色:深度实…...
ShardingSphere分库分表产品介绍
目录 一、ShardingSphere分库分表产品介绍 二、客户端分库分表与服务端分库分表 1、ShardingJDBC客户端分库分表 2、ShardingProxy服务端分库分表 3、ShardingSphere混合部署架构 三、分库分表,能不分就不分! 1、为什么要分库分表? 2、…...
Java经典面试题-多线程打印
threadsynchronized 就好像一个圆圈,A->B->C->A。。。。。 synchronized能够保证多个线程进入实,只用一个线程能进入。 /**多线程交替打印* */ public class Task {private final Object lock new Object();private int count 0;public st…...
FireFox简单设置设置
文章目录 一 设置不显示标签页1原来的样子2新的样子3操作方法 二 设置竖直标签页栏1 效果图2 设置方法 三 设置firefox不提醒更新 一 设置不显示标签页 1原来的样子 2新的样子 3操作方法 地址栏输入 about:config搜索icon,双击选项列表中browserchrome.site icons的值&#…...
Sollong手机——一站式Web3生态解决方案
从定义上讲,Web3公司也属于互联网公司,不过与传统互联网公司相比,他们有一个很明显的特征:他们不断尝试做去中心化的事,一步步将数据和金融的控制权从美联储(央行和金融机构)、苹果(…...
《重生到现代之从零开始的数据结构生活》—— 顺序表1
线性表 线性表:是n个具有相同特性的数据元素的有限序列。 线性表是⼀种在实际中⼴泛使 ⽤的 数据结构,常⻅的线性表有顺序表、链表、栈、队列、字符串等等 线性表在逻辑上是线性结构,也就说是连续的⼀条直线。但是在物理结构上并不⼀定是连…...
Linux 文件类型,目录与路径,文件与目录管理
文件类型 后面的字符表示文件类型标志 普通文件:-(纯文本文件,二进制文件,数据格式文件) 如文本文件、图片、程序文件等。 目录文件:d(directory) 用来存放其他文件或子目录。 设备…...
椭圆曲线密码学(ECC)
一、ECC算法概述 椭圆曲线密码学(Elliptic Curve Cryptography)是基于椭圆曲线数学理论的公钥密码系统,由Neal Koblitz和Victor Miller在1985年独立提出。相比RSA,ECC在相同安全强度下密钥更短(256位ECC ≈ 3072位RSA…...
深入理解JavaScript设计模式之单例模式
目录 什么是单例模式为什么需要单例模式常见应用场景包括 单例模式实现透明单例模式实现不透明单例模式用代理实现单例模式javaScript中的单例模式使用命名空间使用闭包封装私有变量 惰性单例通用的惰性单例 结语 什么是单例模式 单例模式(Singleton Pattern&#…...
《基于Apache Flink的流处理》笔记
思维导图 1-3 章 4-7章 8-11 章 参考资料 源码: https://github.com/streaming-with-flink 博客 https://flink.apache.org/bloghttps://www.ververica.com/blog 聚会及会议 https://flink-forward.orghttps://www.meetup.com/topics/apache-flink https://n…...
3403. 从盒子中找出字典序最大的字符串 I
3403. 从盒子中找出字典序最大的字符串 I 题目链接:3403. 从盒子中找出字典序最大的字符串 I 代码如下: class Solution { public:string answerString(string word, int numFriends) {if (numFriends 1) {return word;}string res;for (int i 0;i &…...
C++.OpenGL (14/64)多光源(Multiple Lights)
多光源(Multiple Lights) 多光源渲染技术概览 #mermaid-svg-3L5e5gGn76TNh7Lq {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-3L5e5gGn76TNh7Lq .error-icon{fill:#552222;}#mermaid-svg-3L5e5gGn76TNh7Lq .erro…...
【Linux】Linux 系统默认的目录及作用说明
博主介绍:✌全网粉丝23W,CSDN博客专家、Java领域优质创作者,掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域✌ 技术范围:SpringBoot、SpringCloud、Vue、SSM、HTML、Nodejs、Python、MySQL、PostgreSQL、大数据、物…...
群晖NAS如何在虚拟机创建飞牛NAS
套件中心下载安装Virtual Machine Manager 创建虚拟机 配置虚拟机 飞牛官网下载 https://iso.liveupdate.fnnas.com/x86_64/trim/fnos-0.9.2-863.iso 群晖NAS如何在虚拟机创建飞牛NAS - 个人信息分享...
华为OD机试-最短木板长度-二分法(A卷,100分)
此题是一个最大化最小值的典型例题, 因为搜索范围是有界的,上界最大木板长度补充的全部木料长度,下界最小木板长度; 即left0,right10^6; 我们可以设置一个候选值x(mid),将木板的长度全部都补充到x,如果成功…...
mac:大模型系列测试
0 MAC 前几天经过学生优惠以及国补17K入手了mac studio,然后这两天亲自测试其模型行运用能力如何,是否支持微调、推理速度等能力。下面进入正文。 1 mac 与 unsloth 按照下面的进行安装以及测试,是可以跑通文章里面的代码。训练速度也是很快的。 注意…...
