PRAGMA SERIALLY_REUSABLE 让Oracle 包中的变量不保存上一次运行的值

本文深入探讨了Oracle中包变量的生命周期及其如何在会话间持久化。介绍了PRAGMA SERIALLY_REUSABLE的作用,即确保包状态仅在单一服务器调用期间有效,避免了会话间的变量污染。通过实例展示了如何在包声明中应用此属性,以维护跨事务的数据而不将其存储在数据库中。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

首要要明白Oracle Package 变量的生命周期是会话。所以如果你在同一会话运行 多次,很有可能变量会使用上一次运行的值。往往这是我们不希望看到。 那么Oracle 提供了 PRAGMA SERIALLY_REUSABLE 来解决这个问题。这命令使得Oracle 只保存One call 的值。 

Packaged public variables and cursors persist for the duration of a session. They can be shared by all subprograms that execute in the environment. They let you maintain data across transactions without storing it in the database.

 

PRAGMA SERIALLY_REUSABLE

Marks a package as serially reusable, if its state is needed only for the duration of one call to the server (for example, an OCI call to the server or a server-to-server remote procedure call). For more information, see Oracle Database Application Developer's Guide - Fundamentals.

 

https://docs.oracle.com/cd/B19306_01/appdev.102/b14261/package_declaration.htm#i35376

 

使用方法:https://docs.oracle.com/cd/B19306_01/B14251_01/adfns_packages.htm#ADFNS009

This example has both a package specification and package body, which are serially reusable.
CREATE OR REPLACE PACKAGE TEST_pkg IS
   PRAGMA SERIALLY_REUSABLE;
   TYPE Str_table_type IS TABLE OF VARCHAR2(200) INDEX BY BINARY_INTEGER;
   Num     NUMBER        := 10;
   Str     VARCHAR2(200) := 'default-init-str';
   Str_tab STR_TABLE_TYPE;
   
    PROCEDURE Print_pkg;
    PROCEDURE Init_and_print_pkg(N NUMBER, V VARCHAR2);
END TEST_pkg;
CREATE OR REPLACE PACKAGE BODY TEST_pkg IS
   -- the body is required to have the pragma because the
  -- specification of this package has the pragma
  PRAGMA SERIALLY_REUSABLE;
   PROCEDURE Print_pkg IS
   BEGIN
      DBMS_OUTPUT.PUT_LINE('num: ' || TEST_pkg.Num);
      DBMS_OUTPUT.PUT_LINE('str: ' || TEST_pkg.Str);
      DBMS_OUTPUT.PUT_LINE('number of table elems: ' || TEST_pkg.Str_tab.Count);
      FOR i IN 1..TEST_pkg.Str_tab.Count LOOP
         DBMS_OUTPUT.PUT_LINE(TEST_pkg.Str_tab(i));
      END LOOP;
   END;
   PROCEDURE Init_and_print_pkg(N NUMBER, V VARCHAR2) IS
   BEGIN
   -- init the package globals
      TEST_pkg.Num := N;
      TEST_pkg.Str := V;
      FOR i IN 1..n LOOP
         TEST_pkg.Str_tab(i) := V || ' ' || i;
   END LOOP;
   -- print the package
   Print_pkg;
   END;
 END TEST_pkg;
<think>好的,用户现在要了解OraclePRAGMA AUTONOMOUS_TRANSACTION。我需要先回顾一下这个知识点的基本内容,确保自己理解正确。 首先,PRAGMA AUTONOMOUS_TRANSACTION是Oracle中用来声明自治事务的编译指令。自治事务的主要特点是独立于主事务,可以单独提交或回滚,影响主事务的状态。这在某些特定场景下非常有用,比如需要在触发器中执行DML操作并提交,而影响主事务的提交或回滚。 用户可能是一位开发人员或者DBA,他们在实际开发中遇到了需要独立事务的情况,比如日志记录。这时候使用自治事务可以避免主事务的锁竞争或者保持日志的独立性。需要确定用户的使用场景,才能给出更具体的建议。 接下来要组织回答的结构,先解释基本概念,然后说明工作原理,接着是使用场景,最后给出示例代码。这样用户能逐步理解。同时要提醒注意事项,比如自治事务的隔离性、资源占用和DDL限制,避免用户误用。 可能用户在使用触发器或存储过程时遇到了事务处理的问题,比如在触发器中无法提交导致主事务被影响,这时候自治事务就派上用场。需要确保用户明白自治事务和主事务的独立性,提交自治事务会影响主事务的状态。 另外,要检查示例代码是否正确,特别是PRAGMA的位置,必须是在存储过程、函数或触发器的声明部分。比如在存储过程中使用AUTONOMOUS_TRANSACTION,确保代码示例准确无误。 用户可能的深层需求是希望了解如何在Oracle中处理需要独立提交的事务,避免事务之间的相互影响。因此,除了基本介绍,还需要强调适用场景和注意事项,帮助他们正确应用。 最后,总结自治事务的优缺点,帮助用户权衡是否在特定情况下使用。比如虽然自治事务提供了独立性,但过度使用可能增加复杂性和资源消耗,需要合理评估。</think>在 Oracle 数据库中,`PRAGMA AUTONOMOUS_TRANSACTION` 是一个编译指令(编译指示),用于声明一个**自治事务**(Autonomous Transaction)。自治事务是一个独立于主事务的子事务,允许在代码块(如存储过程、函数、触发器等)中执行独立的数据库操作(提交或回滚),而会影响主事务的状态。 --- ### **核心概念** 1. **自治事务的特性**: - **独立性**:自治事务与主事务完全隔离。即使主事务回滚,自治事务的提交操作仍然有效,反之亦然。 - **提交/回滚自主性**:自治事务必须显式提交(`COMMIT`)或回滚(`ROLLBACK`),否则会引发错误。 - **用途**:常用于日志记录、审计、错误处理等场景,需保证某些操作受主事务结果的影响。 2. **语法**: 在声明存储过程、函数、触发器或匿名块时,添加以下指令: ```sql PRAGMA AUTONOMOUS_TRANSACTION; ``` --- ### **工作原理** - **主事务**和**自治事务**各自维护独立的事务上下文。 - 自治事务的提交仅影响自身操作,主事务的提交或回滚对自治事务无影响。 ![自治事务示意图](https://img-blog.csdnimg.cn/20201202145717657.png) --- ### **使用场景** 1. **触发器中的独立操作** 例如:在触发器中记录日志,即使主事务回滚,日志仍需保留。 ```sql CREATE OR REPLACE TRIGGER log_audit_trigger AFTER INSERT ON employees FOR EACH ROW DECLARE PRAGMA AUTONOMOUS_TRANSACTION; BEGIN INSERT INTO audit_log (action, timestamp) VALUES ('New employee added', SYSTIMESTAMP); COMMIT; -- 必须显式提交 END; ``` 2. **存储过程/函数中的独立事务** 例如:在存储过程中记录错误信息,受主事务结果影响。 ```sql CREATE OR REPLACE PROCEDURE log_error (p_message VARCHAR2) AS PRAGMA AUTONOMOUS_TRANSACTION; BEGIN INSERT INTO error_log (message, log_time) VALUES (p_message, SYSTIMESTAMP); COMMIT; -- 显式提交 END; ``` 3. **递归调用中的事务控制** 避免主事务与嵌套事务之间的锁冲突。 --- ### **注意事项** 1. **显式提交/回滚** 自治事务必须含 `COMMIT` 或 `ROLLBACK`,否则会报错:`ORA-06519: 检测到活动自治事务,已回退`。 2. **资源隔离性** 自治事务无法直接访问主事务的未提交数据(如未提交的 DML 变更),因为它们在独立的事务上下文中运行。 3. **DDL 操作限制** 在自治事务中执行 DDL(如 `CREATE TABLE`)会隐式提交,可能影响事务逻辑。 4. **性能影响** 频繁创建自治事务会增加数据库资源开销,需合理评估使用场景。 --- ### **示例代码** ```sql -- 自治事务的存储过程示例 CREATE OR REPLACE PROCEDURE autonomous_demo AS PRAGMA AUTONOMOUS_TRANSACTION; BEGIN INSERT INTO test_table (id, name) VALUES (1, 'Autonomous Data'); COMMIT; -- 必须显式提交 END; / -- 主事务中调用自治事务 BEGIN INSERT INTO test_table (id, name) VALUES (2, 'Main Data'); autonomous_demo(); -- 自治事务提交,主事务未提交 ROLLBACK; -- 主事务回滚,但自治事务的数据仍保留 END; ``` --- ### **总结** `PRAGMA AUTONOMOUS_TRANSACTION` 是 Oracle 中实现事务隔离的强有力工具,适用于需独立提交的子操作场景。合理使用可提升灵活性和可靠性,但需注意事务边界和资源管理。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值