0

0

python-oracledb 游标对象详解:生命周期、绑定变量与连接管理

心靈之曲

心靈之曲

发布时间:2025-11-21 11:20:40

|

928人浏览过

|

来源于php中文网

原创

python-oracledb 游标对象详解:生命周期、绑定变量与连接管理

本文深入探讨 `python-oracledb` 中游标对象(cursor)和绑定变量(bind variables)的工作机制。我们将阐明 `cursor.var()` 如何创建客户端 Python 对象以管理绑定变量,并解释数据库会话与游标的生命周期。通过示例代码,纠正关于连接关闭与重开后变量值“持久化”的常见误解,并提供在不同连接间持久化数据的有效策略。

在 python-oracledb 库中,与 Oracle 数据库进行交互的核心组件是连接对象(Connection)和游标对象(Cursor)。理解这些对象的生命周期及其与绑定变量的关系,对于编写健壮、高效的数据库应用程序至关重要。

1. python-oracledb 游标对象与绑定变量概述

在 python-oracledb 中,cursor.var() 方法用于创建绑定变量(Bind Variable)的 Python 客户端表示。这些绑定变量是 Python 应用程序和 Oracle 数据库之间传递数据的桥梁。

  • cursor.var() 的作用: 它在 Python 应用程序的内存中创建一个对象,用于存储待发送到数据库的数据,或接收从数据库返回的数据。例如,host_variable = cursor.var(str) 声明了一个 Python 对象 host_variable,它可以在后续的 cursor.execute() 调用中作为绑定参数使用。
  • 绑定变量的优势: 使用绑定变量可以有效提高 SQL 语句的执行性能(数据库可以重用解析计划),并能有效防止 SQL 注入攻击。

2. Oracle 数据库会话与游标的生命周期

理解数据库会话和游标的生命周期是掌握 python-oracledb 行为的关键。

立即学习Python免费学习笔记(深入)”;

  • 会话的建立与终止: 当您通过 oracledb.connect() 方法成功连接到 Oracle 数据库时,数据库服务器会为该连接创建一个新的用户会话(Session)。这个会话代表了您与数据库的一次交互过程。当您调用 connection.close() 方法时,该数据库会话就会被终止,并且会话期间创建的所有相关资源(包括游标、临时表、PL/SQL 包变量等)都将被释放。
  • 游标的生命周期: Oracle 数据库中的游标是会话级别的资源。无论是显式创建的游标(如 cursor = connection.cursor())还是隐式创建的游标(例如通过 execute() 执行简单查询),它们都与当前的数据库会话绑定。一旦会话结束,所有与之关联的游标及其状态都会失效。
  • 重新连接的影响: 如果您关闭一个连接后,再次调用 oracledb.connect(),这将建立一个全新的数据库连接,并随之创建一个全新的用户会话。这个新会话与之前的会话是完全独立的,不共享任何服务器端的会话资源或状态。

3. 澄清绑定变量的“持久化”误解

一个常见的误解是,当关闭并重新打开数据库连接后,通过 cursor.var() 创建的绑定变量会保留其在数据库中的值。实际上,这通常是由于客户端 Python 对象的生命周期与数据库会话生命周期的混淆所致。

当您执行 host_variable = cursor.var(str) 时,host_variable 是一个在 Python 应用程序内存中创建的普通 Python 对象。host_variable.setvalue(0, 'VALUE') 仅仅是将 'VALUE' 存储在这个 Python 对象内部。这个操作本身并不会将值发送到数据库,也不会在数据库会话中创建任何持久化的状态。

示例代码分析:

考虑以下代码片段,它可能导致“值被保留”的错觉:

阿里云AI平台
阿里云AI平台

阿里云AI平台

下载
import oracledb
import connection_config

# 假设 connection_config 包含数据库连接信息

# 第一次连接与游标
con1 = oracledb.connect(user=connection_config.user, password=connection_config.pw, dsn=connection_config.dsn)
cursor1 = con1.cursor()

# 创建并设置一个客户端Python变量
host_variable = cursor1.var(str)
host_variable.setvalue(0, 'VALUE_FROM_CLIENT')
print(f"第一次连接后,host_variable的值: {host_variable.getvalue()}") # 输出 'VALUE_FROM_CLIENT'

con1.close() # 关闭第一个连接,对应的数据库会话终止

# 第二次连接与游标
con2 = oracledb.connect(user=connection_config.user, password=connection_config.pw, dsn=connection_config.dsn)
cursor2 = con2.cursor()

# 此时,host_variable 仍然是第一次创建的那个 Python 对象。
# 它与新的 cursor2 对象没有任何关联,其内部存储的值在客户端内存中并未改变。
print(f"第二次连接后,host_variable的值: {host_variable.getvalue()}") # 仍然输出 'VALUE_FROM_CLIENT'
con2.close()

上述代码中,host_variable 始终指向同一个 Python 对象。它的值在客户端内存中保持不变,与数据库连接 con1 或 con2 的开关操作无关。这种现象证明的是 Python 变量的客户端生命周期,而非数据库中值的持久性。

正确理解:

为了真正验证数据库会话独立性,应为每个新的游标或连接创建新的绑定变量 Python 对象。

import oracledb
import connection_config

# 第一次连接与游标
con1 = oracledb.connect(user=connection_config.user, password=connection_config.pw, dsn=connection_config.dsn)
cursor1 = con1.cursor()
host_variable1 = cursor1.var(str)
host_variable1.setvalue(0, 'VALUE_FROM_CON1')
print(f"con1: host_variable1 的值: {host_variable1.getvalue()}") # 输出 'VALUE_FROM_CON1'
con1.close()

# 第二次连接与游标
con2 = oracledb.connect(user=connection_config.user, password=connection_config.pw, dsn=connection_config.dsn)
cursor2 = con2.cursor()
host_variable2 = cursor2.var(str) # 为新的游标创建新的绑定变量Python对象

# 此时,host_variable2 尚未被设置过值,也未与数据库进行交互。
# 尝试获取其值,会得到其默认状态(通常是 None 或空字符串,取决于类型和驱动实现)。
print(f"con2: host_variable2 的值: {host_variable2.getvalue()}") # 输出 None 或空字符串
con2.close()

这个修正后的示例清晰地表明,host_variable1 和 host_variable2 是两个独立的 Python 对象,它们各自维护自己的值,互不影响。host_variable2 在创建后没有被显式设置值,因此其 getvalue() 返回的是默认状态,而不是 host_variable1 的值。

4. 客户端与服务器端数据的区别

  • 客户端数据: cursor.var() 创建的 Python 对象及其内部存储的值完全存在于 Python 应用程序的内存中,属于客户端数据。
  • 服务器端数据: 只有当这些绑定变量作为参数传递给 cursor.execute() 方法时,它们的值才会被发送到 Oracle 数据库服务器,参与 SQL 语句的执行。即使值被发送到数据库,除非通过 SQL 语句(如 INSERT、UPDATE)将其显式持久化到数据库表中,否则这些值在数据库会话结束后通常也是短暂的。Oracle PL/SQL 包变量可以提供会话级别的持久性,但它们仍然只存在于当前会话中。

5. 跨连接持久化数据的策略

如果需要在不同的数据库连接或会话之间维护数据,仅仅依赖 cursor.var() 是不够的。以下是几种推荐的策略:

5.1 客户端 Python 变量

将需要持久化的数据存储在 Python 应用程序的普通变量、数据结构(如字典、列表)或自定义对象属性中。这是最简单直接的方法,适用于在应用程序逻辑层面管理数据。

# 示例:在Python应用程序中维护数据
application_state = {
    'user_id': 101,
    'last_login': '2023-10-27',
    'session_token': 'abc123xyz'
}

def perform_db_operation(user_data):
    """使用客户端数据执行数据库操作"""
    con = None
    cursor = None
    try:
        con = oracledb.connect(user=connection_config.user, password=connection_config.pw, dsn=connection_config.dsn)
        cursor = con.cursor()

        # 假设更新用户登录时间
        sql = "UPDATE users SET last_login = :1 WHERE id = :2"
        cursor.execute(sql, (user_data['last_login'], user_data['user_id']))
        con.commit()
        print(f"用户 {user_data['user_id']} 登录时间已更新。")
    except oracledb.Error as e:
        print(f"数据库操作失败: {e}")
    finally:
        if cursor:
            cursor.close()
        if con:
            con.close()

# 在不同时间点,使用客户端维护的数据进行数据库操作
perform_db_operation(application_state)

# 假设应用逻辑更新了数据
application_state['last_login'] = '2023-10-28'
perform_db_operation(application_state)

5.2 Oracle 数据库包变量(会话级)

在 Oracle 数据库中创建 PL/SQL 包,并在其中声明包级变量。这些变量在整个用户会话期间保持其值。但请注意,它们仍然是会话绑定的,一旦会话结束,变量的值就会丢失。

-- Oracle数据库中创建包
CREATE PACKAGE my_session_data_pkg AS
    g_user_preference VARCHAR2(100);
    PROCEDURE set_preference(p_value VARCHAR2);
    FUNCTION get_preference RETURN VARCHAR2;
END my_session_data_pkg;
/

CREATE PACKAGE BODY my_session_data_pkg AS
    PROCEDURE set_preference(p_value VARCHAR2) IS
    BEGIN
        g_user_preference := p_value;
    END;
    FUNCTION get_preference RETURN VARCHAR2 IS
    BEGIN
        RETURN g_user_preference;
    END;
END my_session_data_pkg;
/

在 Python 中使用:

# 在Python中设置和获取包变量
con = oracledb.connect(user=connection_config.user, password=connection_config.pw, dsn=connection_config.dsn)
cursor = con.cursor()

# 设置包变量
cursor.execute("BEGIN my_session_data_pkg.set_preference(:val); END;", val='dark_mode')
con.commit()

# 获取包变量
result_var = cursor.var(str)
cursor.execute("BEGIN :result := my_session_data_pkg.get_preference; END;", result=result_var)
print(f"会话级用户偏好: {result_var.getvalue()}") # 输出 'dark_mode'

cursor.close()
con.close()

# 如果重新连接,该会话变量将重置为NULL或其默认值
con_new = oracledb.connect(user=connection_config.user, password=connection_config.pw

相关文章

python速学教程(入门到精通)
python速学教程(入门到精通)

python怎么学习?python怎么入门?python在哪学?python怎么学才快?不用担心,这里为大家提供了python速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
数据分析工具有哪些
数据分析工具有哪些

数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

1134

2023.10.12

SQL中distinct的用法
SQL中distinct的用法

SQL中distinct的语法是“SELECT DISTINCT column1, column2,...,FROM table_name;”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

340

2023.10.27

SQL中months_between使用方法
SQL中months_between使用方法

在SQL中,MONTHS_BETWEEN 是一个常见的函数,用于计算两个日期之间的月份差。想了解更多SQL的相关内容,可以阅读本专题下面的文章。

381

2024.02.23

SQL出现5120错误解决方法
SQL出现5120错误解决方法

SQL Server错误5120是由于没有足够的权限来访问或操作指定的数据库或文件引起的。想了解更多sql错误的相关内容,可以阅读本专题下面的文章。

2194

2024.03.06

sql procedure语法错误解决方法
sql procedure语法错误解决方法

sql procedure语法错误解决办法:1、仔细检查错误消息;2、检查语法规则;3、检查括号和引号;4、检查变量和参数;5、检查关键字和函数;6、逐步调试;7、参考文档和示例。想了解更多语法错误的相关内容,可以阅读本专题下面的文章。

380

2024.03.06

oracle数据库运行sql方法
oracle数据库运行sql方法

运行sql步骤包括:打开sql plus工具并连接到数据库。在提示符下输入sql语句。按enter键运行该语句。查看结果,错误消息或退出sql plus。想了解更多oracle数据库的相关内容,可以阅读本专题下面的文章。

1703

2024.04.07

sql中where的含义
sql中where的含义

sql中where子句用于从表中过滤数据,它基于指定条件选择特定的行。想了解更多where的相关内容,可以阅读本专题下面的文章。

586

2024.04.29

sql中删除表的语句是什么
sql中删除表的语句是什么

sql中用于删除表的语句是drop table。语法为drop table table_name;该语句将永久删除指定表的表和数据。想了解更多sql的相关内容,可以阅读本专题下面的文章。

440

2024.04.29

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

26

2026.03.13

热门下载

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

精品课程

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

共61课时 | 4.3万人学习

Java 教程
Java 教程

共578课时 | 81.7万人学习

oracle知识库
oracle知识库

共0课时 | 0.6万人学习

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

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