oracle的自定义事物(Procedure)一般是用DBMS_JOBS来调,也可以写个定时器调用。
以下是事物涉及的常用语法,自己看,不解释。。。
CREATE OR REPLACE PROCEDURE PRO_SYNCHRO_A_TO_B /************创建时间XXXX-XX-XX********************* *************把A系统的用户表A_USER数据同步到B系统的用户表B_USER中******************** *********************************/ IS --定义未同步A的A_USER表数据数量 v_a_count NUMBER(15); /****异常定义*****/ v_err_msg VARCHAR2(1024); -- 没有找需要到的数据异常 not_a_user_data_ex EXCEPTION; BEGIN --查询需要导出A_USER用户数量 SELECT COUNT(1) INTO v_a_count FROM A_USER A WHERE NOT EXISTS( SELECT 1 FROM B_USER B WHERE B.ID = A.ID ); IF v_a_count<=0 THEN --没有要导入的数据 RAISE not_a_user_data_ex; END IF; --数据导入 INSERT INTO B_USER(ID,USERNAME,PASSWORD)SELECT ID,USERNAME,PASSWORD FROM A_USER WHERE NOT EXISTS( SELECT 1 FROM B_USER B WHERE B.ID = A.ID ); --写入日志 INSERT INTO T_LOG(EXEC_TIME,EXEC_PRC_NAME,ERROR_MSG,EXEC_ACTION,EXEC_STATUS) VALUES(SYSDATE,'PRO_SYNCHRO_A_TO_B',NULL,'同步A_USER数据#'||v_a_count||'#条','0'); --提交数据 COMMIT; --异常处理 EXCEPTION WHEN not_a_user_data_ex THEN ROLLBACK; --写入日志 INSERT INTO T_LOG(EXEC_TIME,EXEC_PRC_NAME,ERROR_MSG,EXEC_ACTION,EXEC_STATUS) VALUES(SYSDATE,'PRO_SYNCHRO_A_TO_B',NULL,'同步A_USER数据0条','0'); --提交日志数据 COMMIT; WHEN others THEN ROLLBACK; --写入日志 v_err_msg:=SQLERRM; INSERT INTO T_LOG(EXEC_TIME,EXEC_PRC_NAME,ERROR_MSG,EXEC_ACTION,EXEC_STATUS) VALUES(SYSDATE,'PRO_SYNCHRO_A_TO_B',v_err_msg,'同步A_USER失败','1'); --提交日志数据 COMMIT; END;
顺道引网友(http://www.cnblogs.com/ITtangtang/archive/2012/04/23/2466554.html)的例子说明一下事物的一致性和只读属性。
一、事务概念
事务用于保证数据的一致性,它由一组相关的dml语句组成,该组的dml(数据操作语言,增删改,没有查询)语句要么全部成功,要么全部失败。如:网上转账就是典型的要用事务来处理,用于保证数据的一致性。
二、java程序中如何使用事务
在java操作数据库时,为了保证数据的一致性,比如账户操作(1)从一个账户中减掉10$(2)在另一个账户上加入10$,我们看看如何使用事务?
package junit.test; import java.sql.Connection; import java.sql.DriverManager; import java.sql.SQLException; import java.sql.Statement; public class TransationTest { public static void main(String[] args) { Connection conn = null; try { // 1.加载驱动 Class.forName("oracle.jdbc.driver.OracleDriver"); // 2.得到连接 conn = DriverManager.getConnection("jdbc:oracle:thin:@127.0.0.1:1521:orcl", "SCOTT", "scott"); Statement sm = conn.createStatement(); // 从scott的sal中减去100 sm.executeUpdate("update emp set sal=sal-100 where ename='SCOTT'"); int i = 7 / 0; //报java.lang.ArithmeticException: / by zero异常 // 给smith的sal加上100 sm.executeUpdate("update emp set sal=sal+100 where ename='SMITH'"); // 关闭打开的资源 sm.close(); conn.close(); } catch (Exception e) { // 如果发生异常,就回滚 try { conn.rollback(); } catch (SQLException e1) { e1.printStackTrace(); } e.printStackTrace(); } } }
运行,会出现异常,查看数据库,SCOTT 的sal 减了100,但是SMITH 的sal 却不变,很可怕。。。我们怎样才能保证,这两个操作要么同时成功,要么同时失败呢?
package junit.test; import java.sql.Connection; import java.sql.DriverManager; import java.sql.SQLException; import java.sql.Statement; public class TransationTest { public static void main(String[] args) { Connection conn = null; try { // 1.加载驱动 Class.forName("oracle.jdbc.driver.OracleDriver"); // 2.得到连接 conn = DriverManager.getConnection("jdbc:oracle:thin:@127.0.0.1:1521:orcl", "SCOTT", "scott"); // 加入事务处理 conn.setAutoCommit(false);// 设置不能默认提交 Statement sm = conn.createStatement(); // 从scott的sal中减去100 sm.executeUpdate("update emp set sal=sal-100 where ename='SCOTT'"); int i = 7 / 0; // 给smith的sal加上100 sm.executeUpdate("update emp set sal=sal+100 where ename='SMITH'"); // 提交事务 conn.commit(); // 关闭打开的资源 sm.close(); conn.close(); } catch (Exception e) { // 如果发生异常,就回滚 try { conn.rollback(); } catch (SQLException e1) { e1.printStackTrace(); } e.printStackTrace(); } } }
再运行一下,会出现异常,查看数据库,数据没变化。。
三、只读事务
只读事务是指只允许执行查询的操作,而不允许执行任何其它dml操作的事务,使用只读事务可以确保用户只能取得某时间点的数据。假定机票代售点每天18点开始统计今天的销售情况,这时可以使用只读事务。在设置了只读事务后,尽管其它会话可能会提交新的事务,但是只读事务将不会取得最新数据的变化,从而可以保证取得特定时间点的数据信息。
设置只读事务: set transaction read only;
比如有两个用户system、scott各自用sqlplus登陆,操作如下:
第一步:用system用户登陆sqlplus,设置只读事务。
SQL> set transaction read only;
事务处理集。
第二步:用scott用户登陆sqlplus,操作如下:
SQL> select count(*) from emp; --查询emp表的总记录数
COUNT(*)
----------
13
SQL> insert into emp values (7777, 'zhangsan', 'MANAGER', 7782, to_date('1988-02-18', 'yyyy-mm-dd'), 38.38, 45.45, 10); --插入一条记录到emp表
1 row inserted
SQL> select count(*) from emp; --查询emp表的总记录数
COUNT(*)
----------
14
SQL> commit; --提交
Commit complete
第三步:用system用户查询scott.emp表
SQL> select count(*) from scott.emp;
COUNT(*)
----------
13
SQL>