0

0

使用FastAPI与SQLAlchemy查询现有Oracle数据库表

心靈之曲

心靈之曲

发布时间:2025-10-29 14:23:01

|

238人浏览过

|

来源于php中文网

原创

使用FastAPI与SQLAlchemy查询现有Oracle数据库表

本文详细介绍了如何使用fastapi框架结合sqlalchemy orm来查询oracle数据库中已存在的表。核心内容包括配置oracle数据库连接、sqlalchemy模型定义与现有表的映射策略(包括`base.metadata.create_all()`的`checkfirst`默认行为和显式反射机制),以及如何构建fastapi接口提供数据查询服务。文章提供了示例代码和实践建议,帮助开发者高效地实现数据库交互。

1. 环境配置与数据库连接

在使用FastAPI与SQLAlchemy连接Oracle数据库之前,需要确保Python环境中安装了必要的库,并配置好Oracle Instant Client。

1.1. 依赖库安装

首先,安装fastapi、uvicorn(ASGI服务器)、sqlalchemy和cx_Oracle。

pip install fastapi uvicorn sqlalchemy cx_Oracle

1.2. Oracle Instant Client配置

cx_Oracle库需要Oracle Instant Client的支持。请从Oracle官网下载对应操作系统的Instant Client,并解压。然后,在Python代码中通过cx_Oracle.init_oracle_client()指定其lib_dir路径。

import cx_Oracle
# 请将此路径替换为您的Oracle Instant Client的实际lib_dir路径
# 例如:Windows系统下 r"C:\oracle\instantclient_19_12"
# Linux/macOS系统下 "/opt/oracle/instantclient_19_12"
cx_Oracle.init_oracle_client(lib_dir=r"E:\instantclient-basic-windows.x64-12.1.0.2.0\instantclient_12_1")

1.3. 数据库引擎与会话管理

使用SQLAlchemy的create_engine函数创建数据库连接引擎,并配置会话工厂sessionmaker。Oracle数据库的连接字符串通常格式为oracle+cx_oracle://user:password@host:port/service_name或oracle+cx_oracle://user:password@tnsname。

from sqlalchemy import create_engine
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.orm import sessionmaker, Session

# 替换为您的Oracle数据库连接信息
# 示例:oracle+cx_oracle://用户名:密码@主机名:端口号/服务名
my_database_connection = "oracle+cx_oracle://super:your_password@localhost:1521/db"

engine = create_engine(my_database_connection)
SessionLocal = sessionmaker(autocommit=False, autoflush=False, bind=engine)
Base = declarative_base()

2. SQLAlchemy模型定义与现有表映射

当数据库中已存在表时,SQLAlchemy提供了多种方式来定义模型并映射到这些表。

2.1. 默认行为:Base.metadata.create_all()的checkfirst机制

许多开发者在处理现有表时,会误以为Base.metadata.create_all(bind=engine)会尝试重新创建表导致错误。然而,SQLAlchemy的MetaData.create_all()方法默认包含一个checkfirst=True的参数。这意味着在尝试创建表之前,它会检查数据库中是否已存在同名表。如果表已存在,create_all()将忽略该表,不会尝试重新创建,也不会引发错误。因此,即使您的表已经存在,使用此行代码通常也是安全的。

以下代码示例展示了这种默认行为,它定义了一个名为table1的模型,并尝试通过create_all进行映射。由于checkfirst=True是默认值,如果table1已存在,此操作将无害地通过。

from fastapi import Depends, FastAPI
from sqlalchemy import Column, Integer, String

# ... (前文的cx_Oracle初始化、engine和SessionLocal定义) ...

class table1(Base):
    __tablename__ = "table1" # 映射到数据库中名为"table1"的表
    # 定义表的列,注意主键是必须的,这里假设CN是主键或唯一标识
    # 如果没有显式主键,SQLAlchemy可能无法正确操作
    CN = Column(String(length=256), primary_key=True) # 假设CN是主键
    NAME = Column(String(length=40))
    EMAIL = Column(String(length=20))

# 即使table1已存在,此行代码在默认checkfirst=True的情况下也不会报错
# 它会检查表是否存在,如果存在则跳过创建
Base.metadata.create_all(bind=engine)

app = FastAPI()

def get_db():
    db = SessionLocal()
    try:
        yield db
    finally:
        db.close()

@app.get("/table_api/")
def read_table1_data(db: Session = Depends(get_db)):
    """
    查询table1中的所有数据
    """
    cn_information = db.query(table1).all()
    return cn_information

2.2. 显式映射:使用反射机制处理现有表

新快购物系统
新快购物系统

新快购物系统是集合目前网络所有购物系统为参考而开发,不管从速度还是安全我们都努力做到最好,此版虽为免费版但是功能齐全,无任何错误,特点有:专业的、全面的电子商务解决方案,使您可以轻松实现网上销售;自助式开放性的数据平台,为您提供充满个性化的设计空间;功能全面、操作简单的远程管理系统,让您在家中也可实现正常销售管理;严谨实用的全新商品数据库,便于查询搜索您的商品。

下载

虽然create_all的默认行为通常足够,但在某些情况下,您可能希望更显式地声明模型是基于数据库中现有表反射而来,而不是由ORM管理其创建。SQLAlchemy提供了反射(Reflection)机制来实现这一点。

您可以利用sqlalchemy.schema.Table和autoload_with参数来显式地从数据库反射表的结构。

from sqlalchemy import Table, MetaData
from sqlalchemy.orm import declarative_base

# ... (前文的cx_Oracle初始化、engine和SessionLocal定义) ...

Base = declarative_base()
metadata = MetaData() # 创建一个MetaData实例

# 显式反射现有表table1
# autoload_with=engine 会让SQLAlchemy连接数据库并自动加载表的列信息
class table1_reflected(Base):
    __table__ = Table("table1", metadata, autoload_with=engine)
    # 可以在这里添加额外的ORM属性或关系,但列定义会从数据库中反射

# 注意:使用反射机制时,通常不需要调用Base.metadata.create_all()
# 因为我们是基于已存在的表进行映射,而不是创建新表。
# 如果您在同一个应用中混合使用了创建和反射,请谨慎管理MetaData实例。

app = FastAPI()

def get_db():
    db = SessionLocal()
    try:
        yield db
    finally:
        db.close()

@app.get("/table_api_reflected/")
def read_table1_reflected_data(db: Session = Depends(get_db)):
    """
    查询通过反射机制映射的table1数据
    """
    # 查询时使用反射后的模型类
    cn_information = db.query(table1_reflected).all()
    return cn_information

在这个例子中,table1_reflected模型通过__table__ = Table("table1", metadata, autoload_with=engine)显式地从数据库中反射了table1的结构。这种方式更清晰地表明模型是基于一个已存在的数据库表。

3. 构建FastAPI查询接口

FastAPI通过依赖注入(Depends)管理数据库会话,使得接口的编写更加简洁和模块化。

3.1. 数据库会话依赖

定义一个get_db函数作为FastAPI的依赖项,用于在每个请求中获取并管理数据库会话。

# ... (上述SQLAlchemy模型定义) ...

def get_db():
    db = SessionLocal()
    try:
        yield db # 将数据库会话传递给路由函数
    finally:
        db.close() # 请求结束后关闭会话

3.2. 定义API端点

使用@app.get()装饰器定义HTTP GET请求的API端点,并通过Depends(get_db)注入数据库会话。

# ... (上述FastAPI应用实例和get_db函数) ...

@app.get("/table_api/")
def read_table_data(db: Session = Depends(get_db)):
    """
    查询table1中的所有数据并返回
    """
    # 使用SQLAlchemy会话执行查询
    cn_information = db.query(table1).all() # 使用之前定义的table1模型
    return cn_information

运行FastAPI应用:

uvicorn main:app --reload

然后访问 http://127.0.0.1:8000/table_api/ 即可看到查询结果。

4. 注意事项与最佳实践

  • Oracle Instant Client路径: 务必确保cx_Oracle.init_oracle_client(lib_dir=...)中的路径正确,否则会导致连接失败。
  • 数据库连接字符串安全: 在生产环境中,不应将数据库用户名和密码硬编码在代码中。应使用环境变量、配置文件或Secrets管理工具来存储和加载敏感信息。
  • 主键定义: SQLAlchemy ORM要求模型至少有一个主键。如果您的表没有显式主键,或者您没有在模型中正确定义主键,可能会导致ORM操作(如查询、更新、删除)出现问题。在反射模式下,SQLAlchemy通常能自动识别主键。
  • 何时选择不同映射策略:
    • Base.metadata.create_all() (默认checkfirst=True): 适用于应用程序既可能创建新表,也可能连接现有表的情况。它提供了一种便捷的“确保表存在”机制。
    • 显式反射 (__table__ = Table(...)): 当您明确知道表已存在,并且希望模型严格按照数据库现有结构定义时,反射是更清晰的选择。它避免了任何潜在的“创建”行为,并能更好地与数据库管理员协作。
  • 错误处理: 在实际应用中,应为数据库连接、查询等操作添加适当的错误处理机制,例如使用try-except块捕获sqlalchemy.exc.DBAPIError等异常。
  • 性能优化: 对于大数据量查询,考虑使用分页、索引优化或异步数据库驱动(如asyncpg配合databases库)来提升性能。

总结

通过本文的指导,您应该能够熟练地使用FastAPI和SQLAlchemy来连接Oracle数据库并查询现有表。理解Base.metadata.create_all()的checkfirst默认行为,以及掌握显式反射机制,是高效处理现有数据库模式的关键。结合FastAPI的依赖注入,可以构建出结构清晰、易于维护的API服务。在实际开发中,请务必关注安全性、错误处理和性能优化等最佳实践。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
Python FastAPI异步API开发_Python怎么用FastAPI构建异步API
Python FastAPI异步API开发_Python怎么用FastAPI构建异步API

Python FastAPI 异步开发利用 async/await 关键字,通过定义异步视图函数、使用异步数据库库 (如 databases)、异步 HTTP 客户端 (如 httpx),并结合后台任务队列(如 Celery)和异步依赖项,实现高效的 I/O 密集型 API,显著提升吞吐量和响应速度,尤其适用于处理数据库查询、网络请求等耗时操作,无需阻塞主线程。

28

2025.12.22

Python 微服务架构与 FastAPI 框架
Python 微服务架构与 FastAPI 框架

本专题系统讲解 Python 微服务架构设计与 FastAPI 框架应用,涵盖 FastAPI 的快速开发、路由与依赖注入、数据模型验证、API 文档自动生成、OAuth2 与 JWT 身份验证、异步支持、部署与扩展等。通过实际案例,帮助学习者掌握 使用 FastAPI 构建高效、可扩展的微服务应用,提高服务响应速度与系统可维护性。

249

2026.02.06

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

658

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

219

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1560

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

645

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

1088

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

1042

2024.04.29

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

24

2026.02.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
SQL 教程
SQL 教程

共61课时 | 4.1万人学习

Java 教程
Java 教程

共578课时 | 75.1万人学习

oracle知识库
oracle知识库

共0课时 | 0.6万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号