ORM框架详解:为什么不直接写SQL?

想象一下,你正在开发一个小型的在线书店应用。你需要存储书籍信息、用户数据和订单记录。作为一个初学者,你可能会想:“我已经学会了SQL,为什么还要使用ORM框架呢?直接写SQL语句不是更简单、更直接吗?”

如果你有类似的疑问,那么恭喜你!你已经踏上了深入理解数据库交互的journey。在这篇文章中,我们将一起探索ORM框架的世界,了解它为什么存在,以及它如何能够提升你的开发效率和代码质量。
image.png

目录
    • 什么是ORM框架?
    • 为什么需要ORM框架?
    • ORM vs 直接SQL:一个实际例子
      • 直接使用SQL
      • 使用ORM(以SQLAlchemy为例)
    • ORM的优势
    • ORM的潜在缺点
    • 常见的ORM框架
    • 如何选择合适的ORM框架
    • 实际应用:使用ORM构建在线书店后端
    • 结论

什么是ORM框架?

ORM是"Object-Relational Mapping"的缩写,中文通常翻译为"对象关系映射"。这个术语听起来可能有点抽象,让我们通过一个简单的比喻来理解它:

想象你是一位翻译官,你的工作是在两种完全不同的语言之间进行翻译。在编程世界中,ORM就像这样一位翻译官,它在面向对象的编程语言(如Java、Python、C#等)和关系型数据库(如MySQL、PostgreSQL、Oracle等)之间进行"翻译"。

具体来说,ORM框架允许你:

  1. 使用面向对象的方式来操作数据库
  2. 将数据库表映射到编程语言中的类
  3. 将表中的记录映射到类的实例(对象)
  4. 将表的字段映射到对象的属性

通过这种映射,你可以使用熟悉的面向对象编程(OOP)概念和语法来进行数据库操作,而不需要直接编写SQL语句。
image.png

为什么需要ORM框架?

在回答这个问题之前,让我们先思考一下直接使用SQL可能会遇到的一些挑战:

  1. 语言不匹配:SQL是一种声明式语言,而大多数编程语言是命令式的。这种范式的差异可能导致代码的不一致性和复杂性。

  2. 代码重复:对于常见的CRUD(创建、读取、更新、删除)操作,你可能会发现自己在不同的地方重复编写相似的SQL语句。

  3. 安全性问题:直接拼接SQL字符串容易导致SQL注入攻击,需要额外的注意和处理。

  4. 数据库依赖:直接编写SQL会使你的代码与特定的数据库系统紧密耦合,难以切换到其他数据库。

  5. 面向对象的不匹配:在面向对象的程序中,你需要手动将SQL查询结果转换为对象,这个过程可能很繁琐。
    image.png

ORM框架的出现就是为了解决这些问题。它提供了一个抽象层,使得开发者可以用面向对象的方式来操作数据库,从而提高开发效率,减少错误,并使代码更易于维护。

ORM vs 直接SQL:一个实际例子

让我们通过一个具体的例子来比较使用ORM和直接写SQL的区别。假设我们要实现earlier提到的在线书店应用中的一个功能:根据作者名称查询书籍并更新价格。
image.png

直接使用SQL
import mysql.connector# 连接数据库
conn = mysql.connector.connect(host="localhost",user="yourusername",password="yourpassword",database="bookstore"
)
cursor = conn.cursor()# 查询书籍
author_name = "J.K. Rowling"
query = "SELECT id, title, price FROM books WHERE author = %s"
cursor.execute(query, (author_name,))
books = cursor.fetchall()# 更新价格
for book in books:book_id, title, current_price = booknew_price = current_price * 1.1  # 提高10%的价格update_query = "UPDATE books SET price = %s WHERE id = %s"cursor.execute(update_query, (new_price, book_id))print(f"Updated price for '{title}' from {current_price} to {new_price}")# 提交事务并关闭连接
conn.commit()
cursor.close()
conn.close()
使用ORM(以SQLAlchemy为例)
from sqlalchemy import create_engine, Column, Integer, String, Float
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.orm import sessionmaker# 创建数据库引擎
engine = create_engine('mysql://yourusername:yourpassword@localhost/bookstore')
Base = declarative_base()# 定义Book模型
class Book(Base):__tablename__ = 'books'id = Column(Integer, primary_key=True)title = Column(String(100))author = Column(String(50))price = Column(Float)# 创建会话
Session = sessionmaker(bind=engine)
session = Session()# 查询和更新书籍
author_name = "J.K. Rowling"
books = session.query(Book).filter_by(author=author_name).all()for book in books:book.price *= 1.1  # 提高10%的价格print(f"Updated price for '{book.title}' from {book.price/1.1:.2f} to {book.price:.2f}")# 提交事务
session.commit()
session.close()

通过比较这两段代码,我们可以看到使用ORM带来的一些明显优势:

  1. 代码简洁性:ORM版本的代码更加简洁,不需要手动编写SQL语句。

  2. 面向对象的操作:在ORM版本中,我们直接操作Book对象,这与面向对象编程的思想更加一致。

  3. 安全性:ORM自动处理参数化查询,减少SQL注入的风险。

  4. 可读性:ORM代码更接近自然语言描述,更容易理解代码的意图。

  5. 数据库无关性:如果需要切换到不同的数据库系统,只需要更改连接字符串,而不需要重写SQL语句。

ORM的优势

image.png

通过上面的例子,我们已经看到了ORM的一些优点。让我们更系统地总结一下ORM框架的主要优势:

  1. 生产力提升

    • 减少样板代码:ORM自动处理了许多底层的数据库操作,如连接管理、SQL生成等。
    • 快速开发:使用ORM可以更快地构建数据模型和执行常见的数据库操作。
  2. 面向对象的优雅

    • 自然的编程模型:ORM允许你用面向对象的方式思考和操作数据,这与大多数现代编程语言的范式一致。
    • 继承和多态:可以利用面向对象的特性来设计更灵活、可扩展的数据模型。
  3. 可维护性

    • 集中的数据模型定义:数据模型通常定义在一个地方,便于管理和修改。
    • 减少重复代码:常见的数据库操作被抽象化,减少了代码重复。
  4. 数据库无关性

    • 易于切换数据库:大多数ORM支持多种数据库后端,切换数据库只需要更改配置。
    • 跨数据库的一致API:无论底层使用什么数据库,你都可以使用相同的API进行操作。
  5. 安全性

    • 参数化查询:ORM自动使用参数化查询,大大降低SQL注入的风险。
    • 数据验证:许多ORM框架提供了数据验证功能,可以在数据进入数据库之前进行检查。
  6. 性能优化

    • 延迟加载:ORM可以智能地决定何时从数据库加载数据,避免不必要的查询。
    • 缓存:一些ORM框架提供查询缓存功能,可以提高反复查询的性能。
  7. 版本控制和迁移

    • 数据库迁移:许多ORM框架提供了数据库迁移工具,使得管理数据库schema的变更变得更加容易。
    • 版本控制:数据模型可以像其他代码一样进行版本控制。
  8. 测试友好

    • 易于模拟:使用ORM,你可以更容易地模拟数据库操作,便于单元测试。
    • 内存数据库:许多ORM支持内存数据库,可以加速测试过程。

ORM的潜在缺点

尽管ORM框架带来了许多优势,但它也不是没有缺点。了解这些潜在的问题对于正确使用ORM非常重要:

  1. 性能开销

    • 额外的抽象层:ORM在应用程序和数据库之间增加了一个抽象层,这可能导致一些性能开销。
    • 可能生成次优的SQL:在复杂查询场景下,ORM生成的SQL可能不如手写的SQL优化。
  2. 学习曲线

    • 新的概念:使用ORM需要学习新的概念和API,对于初学者来说可能有一定难度。
    • 配置复杂性:某些ORM框架的配置可能比较复杂,需要时间来掌握。
  3. "漏抽象"问题

    • 无法完全隐藏SQL:在某些复杂查询场景下,你可能还是需要编写原生SQL或了解底层的SQL知识。
    • 特定数据库功能:某些数据库特有的高级功能可能无法通过ORM直接使用。
  4. 过度使用的风险

    • N+1查询问题:如果不小心,很容易导致N+1查询问题,影响性能。
    • 加载过多数据:如果不正确使用,可能会从数据库加载不必要的数据。
  5. 调试困难

    • SQL不可见:由于SQL是动态生成的,调试复杂查询可能会变得困难。
    • 错误信息不清晰:ORM的错误信息有时可能不如直接的SQL错误信息清晰。
  6. 版本兼容性

    • ORM更新可能带来不兼容:ORM框架的主要版本更新可能需要修改现有代码。
    • 数据库驱动兼容性:ORM可能对特定版本的数据库驱动有依赖。
      image.png

尽管存在这些潜在的缺点,但对于大多数应用程序来说,ORM的优势仍然远大于缺点。关键是要理解这些限制,并在适当的时候做出权衡。

常见的ORM框架

不同的编程语言通常有其流行的ORM框架。以下是一些主流编程语言中常用的ORM框架:

  1. Python

    • SQLAlchemy:功能强大、灵活性高的ORM框架
    • Django ORM:Django web框架自带的ORM
    • Peewee:轻量级、简单易用的ORM
  2. Java

    • Hibernate:最流行的Java ORM框架之一
    • JPA (Java Persistence API):Java EE的ORM标准
    • MyBatis:一种"半自动"的ORM框架
  3. C#/.NET

    • Entity Framework:微软官方的ORM框架
    • NHibernate:Hibernate的.NET移植版
    • Dapper:一个轻量级的ORM框架
  4. Ruby

    • Active Record:Ruby on Rails框架中使用的ORM
    • Sequel:一个独立的Ruby ORM
  5. JavaScript/Node.js

    • Sequelize:支持多种数据库的ORM
    • TypeORM:支持TypeScript的ORM
    • Mongoose:专门用于MongoDB的ODM(对象文档映射)
  6. PHP

    • Doctrine:受Hibernate启发的PHP ORM
    • Eloquent:Laravel框架中使用的ORM
      image.png

每个框架都每个框架都有其独特的特性和优势,选择哪一个通常取决于你的具体需求、项目规模、以及个人或团队的偏好。

如何选择合适的ORM框架

选择合适的ORM框架对于项目的成功至关重要。以下是一些选择ORM框架时需要考虑的因素:

  1. 语言和生态系统兼容性

    • 确保ORM框架与你的主要编程语言有良好的集成。
    • 考虑框架在该语言生态系统中的地位和受欢迎程度。
  2. 学习曲线

    • 评估你和你的团队学习新框架所需的时间。
    • 考虑框架的文档质量和社区支持。
  3. 性能

    • 研究框架在处理大量数据时的性能表现。
    • 考虑框架是否提供性能优化工具,如缓存机制。
  4. 功能集

    • 确保框架支持你需要的所有数据库操作。
    • 检查是否支持高级功能,如复杂查询、事务管理等。
  5. 数据库支持

    • 确保框架支持你计划使用的数据库系统。
    • 如果你需要支持多个数据库,检查框架的跨数据库能力。
  6. 可扩展性

    • 考虑框架是否能够随着项目的增长而扩展。
    • 检查是否支持分布式系统或微服务架构。
  7. 社区和维护

    • 查看框架的GitHub星数、贡献者数量等指标。
    • 检查最近的更新频率,确保框架仍在积极维护。
  8. 企业支持

    • 如果是企业级项目,考虑是否有商业支持选项。
  9. 与其他工具的集成

    • 检查框架是否能与你使用的其他开发工具良好集成。
  10. 测试支持

*   考虑框架是否提供良好的测试支持,如易于模拟的API。  ![image.png](https://img-blog.csdnimg.cn/img_convert/c6879f6594cb0052dbba2e68830d96eb.png)

通过仔细权衡这些因素,你可以为你的项目选择最合适的ORM框架。记住,没有一个框架是完美的或适合所有场景的,关键是找到最适合你特定需求的解决方案。

实际应用:使用ORM构建在线书店后端

为了更好地理解ORM在实际项目中的应用,让我们用Python和SQLAlchemy来构建一个简单的在线书店后端。这个例子将展示如何定义模型、执行查询、以及处理关系。

首先,我们需要安装必要的依赖:

pip install sqlalchemy

然后,我们可以开始编写我们的代码:

from sqlalchemy import create_engine, Column, Integer, String, Float, ForeignKey
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.orm import sessionmaker, relationship# 创建数据库引擎
engine = create_engine('sqlite:///bookstore.db', echo=True)
Base = declarative_base()# 定义模型
class Author(Base):__tablename__ = 'authors'id = Column(Integer, primary_key=True)name = Column(String(100), nullable=False)books = relationship("Book", back_populates="author")class Book(Base):__tablename__ = 'books'id = Column(Integer, primary_key=True)title = Column(String(100), nullable=False)price = Column(Float, nullable=False)author_id = Column(Integer, ForeignKey('authors.id'))author = relationship("Author", back_populates="books")# 创建表
Base.metadata.create_all(engine)# 创建会话
Session = sessionmaker(bind=engine)
session = Session()# 添加数据
author1 = Author(name="J.K. Rowling")
book1 = Book(title="Harry Potter and the Philosopher's Stone", price=19.99, author=author1)
book2 = Book(title="Harry Potter and the Chamber of Secrets", price=21.99, author=author1)session.add(author1)
session.add_all([book1, book2])
session.commit()# 查询数据
authors = session.query(Author).all()
for author in authors:print(f"Author: {author.name}")for book in author.books:print(f"  - {book.title} (${book.price:.2f})")# 更新数据
book_to_update = session.query(Book).filter_by(title="Harry Potter and the Philosopher's Stone").first()
if book_to_update:book_to_update.price = 24.99session.commit()print(f"Updated price of '{book_to_update.title}' to ${book_to_update.price:.2f}")# 删除数据
book_to_delete = session.query(Book).filter_by(title="Harry Potter and the Chamber of Secrets").first()
if book_to_delete:session.delete(book_to_delete)session.commit()print(f"Deleted '{book_to_delete.title}'")session.close()

这个例子展示了如何:

  1. 定义数据模型(Author和Book)
  2. 建立模型之间的关系
  3. 创建数据库表
  4. 添加、查询、更新和删除数据

通过这个简单的例子,我们可以看到ORM如何简化数据库操作,使得代码更加直观和面向对象。

结论

ORM框架为开发者提供了一种强大的工具,使得数据库操作变得更加简单、直观和安全。虽然直接编写SQL在某些情况下可能更简单或更高效,但ORM带来的好处通常超过了其潜在的缺点。

ORM的主要优势包括:

  • 提高开发效率
  • 增强代码的可维护性
  • 提供数据库无关性
  • 增强应用程序的安全性
  • 允许利用面向对象编程的优势

然而,使用ORM也需要权衡一些因素:

  • 可能带来一定的性能开销
  • 存在学习曲线
  • 在某些复杂查询场景可能不如直接SQL灵活

对于大多数现代web应用程序和企业系统来说,ORM已经成为了标准工具。它不仅简化了开发过程,还提高了代码质量和可维护性。然而,像所有工具一样,ORM也不是万能的。理解ORM的工作原理、优势和局限性,才能在适当的场景下做出正确的选择。

最后,记住:ORM和SQL并不是非此即彼的选择。在实际项目中,你可能会发现将ORM与原生SQL结合使用是最佳实践。大多数ORM框架都提供了执行原生SQL的能力,让你能够在需要时利用SQL的全部功能。
image.png

无论你选择使用ORM还是直接编写SQL,重要的是要理解底层的数据库原理,这样你才能做出明智的决策,并在需要时进行优化。持续学习和实践将帮助你在不同场景下选择最合适的工具和方法。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/891171.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

合合信息亮相CSIG AI可信论坛,全面拆解AI视觉内容安全的“终极防线”

合合信息亮相CSIG AI可信论坛,全面拆解视觉内容安全的“终极防线”! 🐯 AI伪造泛滥,我们还能相信“眼见为实”吗? 近期,由中国图象图形学学会主办的CSIG青年科学家会议 AI可信论坛在杭州成功举办。本次论…

Linux实验报告9-进程管理

目录 一:实验目的 二:实验内容 (1)列出当前系统中的所有进程,如何观察进程的优先级? (2)查看当前终端运行的 bash 进程的 PID,在当前终端启动 vim 编辑器并让其在后台执行,然后列出在当前终端中执行的进程的家族树。 (3)请自行挂载U盘或光盘,然后…

17爬虫:关于DrissionPage相关内容的学习01

概述 前面我们已经大致了解了selenium的用法,DerssionPage同selenium一样,也是一个基于Python的网页自动化工具。 DrissionPage既可以实现网页的自动化操作,也能够实现收发数据包,也可以把两者的功能合二为一。 DressionPage的…

【Unity3D】Jobs、Burst并行计算裁剪Texture3D物体

版本:Unity2019.4.0f1 PackageManager下载Burst插件(1.2.3版本) 利用如下代码,生成一个Texture3D资源,它只能脚本生成,是一个32*32*32的立方体,导出路径记得改下,不然报错。 using UnityEditor; using Uni…

最短路径-Dijkstra 算法

前言 迪杰斯特拉算法是由荷兰计算机科学家狄克斯特拉于1959 年提出的,因此又叫狄克斯特拉算法。是从一个顶点到其余各顶点的最短路径算法,解决的是有向图中最短路径问题。迪杰斯特拉算法主要特点是以起始点为中心向外层层扩展,直到扩展到终点…

ESP32 I2S音频总线学习笔记(一):初识I2S通信与配置基础

文章目录 简介为什么需要I2S?关于音频信号采样率分辨率音频声道 怎样使用I2S传输音频?位时钟BCLK字时钟WS串行数据SD I2S传输模型I2S通信格式I2S格式左对齐格式右对齐格式 i2s基本配置i2s 底层API加载I2S驱动设置I2S使用的引脚I2S读取数据I2S发送数据卸载…

Eclipse中引入NS3项目

参考资料: 博主:深度不睡觉 NS3的3.36版本将Eclipse作IDE_ns3使用eclipse-CSDN博客 从1.2安装eclipse开始 其中参考教程中省略的几点: 1.下载解压tar包 mkdir /Tools/Eclipse/EclipseTool # 新建目录 tar -zxvf /path/to/eclipse-cpp-20…

机器学习周报-TCN文献阅读

文章目录 摘要Abstract 1 TCN通用架构1.1 序列建模任务描述1.2 因果卷积(Causal Convolutions)1.3 扩张卷积(Dilated Convolutions)1.4 残差连接(Residual Connections) 2 TCN vs RNN3 TCN缺点4 代码4.1 TC…

Quartz任务调度框架实现任务动态执行

说明:之前使用Quartz,都是写好Job,指定一个时间点,到点执行。最近有个需求,需要根据前端用户设置的时间点去执行,也就是说任务执行的时间点是动态变化的。本文介绍如何用Quartz任务调度框架实现任务动态执行…

Spring-kafka快速Demo示例

使用Spring-Kafka快速发送/接受Kafka消息示例代码&#xff0c;项目结构是最基础的SpringBoot结构&#xff0c;提前安装好Kafka&#xff0c;确保Kafka已经正确启动 pom.xml&#xff0c;根据个人情况更换springboot、java版本等 <?xml version"1.0" encoding&qu…

【C++】B2079 求出 e 的值

博客主页&#xff1a; [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: C 文章目录 &#x1f4af;前言&#x1f4af;题目介绍输入格式输出格式输入输出样例说明/提示 &#x1f4af;实现方法一&#xff1a;单层 for 循环计算代码实现运行逻辑解析优点不足 &#x1f4af;实现方法二&#xff…

STM32配合可编程加密芯片SMEC88ST的防抄板加密方案设计

SMEC88ST SDK卡发包下载 目前市场上很多嵌入式产品方案都是可以破解复制的&#xff0c;主要是因为方案主芯片不具备防破解的功能&#xff0c;这就导致开发者投入大量精力、财力开发的新产品一上市就被别人复制&#xff0c;到市场上的只能以价格竞争&#xff0c;最后工厂复制的产…

精准识别花生豆:基于EfficientNetB0的深度学习检测与分类项目

精准检测花生豆&#xff1a;基于EfficientNet的深度学习分类项目 在现代农业生产中&#xff0c;作物的质量检测和分类是确保产品质量的重要环节。针对花生豆的检测与分类需求&#xff0c;我们开发了一套基于深度学习的解决方案&#xff0c;利用EfficientNetB0模型实现高效、准…

MarkItDown的使用(将Word、Excel、PDF等转换为Markdown格式)

MarkItDown的使用&#xff08;将Word、Excel、PDF等转换为Markdown格式&#xff09; 本文目录&#xff1a; 零、时光宝盒&#x1f33b; 一、简介 二、安装 三、使用方法 3.1、使用命令行形式 3.2、用 Python 调用 四、总结 五、参考资料 零、时光宝盒&#x1f33b; &a…

Qanything 2.0源码解析系列6 PDF解析逻辑

Qanything 2.0源码解析系列6: PDF解析逻辑 type: Post status: Published date: 2024/12/04 summary: 深入剖析Qanything是如何拆解PDF的,核心是pdf转markdown category: 技术分享 原文:www.feifeixu.top 😀 前言: 在前面的文章中探究了图片是怎么进行解析的,这篇文章对…

【Agent】Chatbot、Copilot与Agent如何帮助我们的提升效率?

人工智能&#xff08;AI&#xff09;技术的迅猛发展正在深刻改变我们的生活和工作方式。你是否曾想过&#xff0c;未来的工作场景会是什么样子&#xff1f;AI的崛起不仅仅是科技的进步&#xff0c;更是我们生活方式的革命。今天&#xff0c;我们将深入探讨三种主要的AI能力&…

如何使用python读写游戏内存以及使用特征码匹配基址

一.读写内存所需的基本参数 接下来我将使用GTA5游戏举例 1.通过进程名称获取进程pid from psutil import process_iterdef get_process_id_by_name(process_name):for process in process_iter(["pid", "name"]):if process.info["name"] pr…

简述css中z-index的作用?如何用定位使用?

z-index是一个css属性&#xff0c;用于控制元素的堆叠顺序&#xff0c; 如何使用定位用index 1、position&#xff1a;relative&#xff1b; z-index&#xff1b; 相对于自己来定位的&#xff0c;可以根据top&#xff0c;bottom&#xff0c;right&#xff0c;left&#xff…

CCNP_SEC_ASA 第六天作业

实验需求&#xff1a; 为保障内部用户能够访问Internet&#xff0c;请把10.1.1.0/24网络动态转换到外部地址池202.100.1.100-202.100.1.200&#xff0c;如果地址池耗尽后&#xff0c;PAT到Outside接口 提示&#xff1a;需要看到如下输出信息 Inside#telnet 202.100.1.1 Trying …

计算机网络 (13)信道复用技术

前言 计算机网络中的信道复用技术是一种提高网络资源利用率的关键技术。它允许在一条物理信道上同时传输多个用户的信号&#xff0c;从而提高了信道的传输效率和带宽利用率。 一、信道复用技术的定义 信道复用&#xff08;Multiplexing&#xff09;就是在一条传输媒体上同时传输…