开发者

Is there a way to determine if JDBC transaction is in effect?

开发者 https://www.devze.com 2023-02-27 10:46 出处:网络
I am calling several SQLServer 2008 stored procedures through JDBC. I would like all these proc calls to be a part of the same transaction.

I am calling several SQLServer 2008 stored procedures through JDBC.

I would like all these proc calls to be a part of the same transaction.

On java side I have the equivalent of the following:

con.setAutoCommit( false );
boolean hasFailed = true;

try
{
  ... call PROC_1

  ... call PROC_2

  con.commit( );
  hasFailed = false;
}
finally
{
  if ( hasFailed )
  {
    con.rollback( );
  }
  con.setAutoCommit( true );
}

In PROC_1 I have a transaction guard that only starts a new transaction if there is no transaction in effect.

The guard is implemented by checking the value of @@TRANCOUNT

declare @owns_transaction int = 0

begin try

  if @@TRANCOUNT = 0
  begin
    begin transaction
    set @owns_transaction = 1
  end

  ... do work

  if @owns_transaction = 1
  begin
    commit transaction
    set @owns_transaction = 0
  end
end try
begin catch
  if @owns_transaction = 1
  begin
    rollback transaction
    set @owns_transaction = 0
  end
  ... handle error
end catch

However, when I enter PROC_1 the @@TRANCOUNT is still 0, so it starts a new transaction and I think it may have some dire consequences later on.

I've tried to use XACT_STATE(), but it also returns a result of 0, which means 'There is no active user transaction for the current request.', and according to this I have to start a manual transaction as well.

Am I doing something wrong?

BTW, I think SQLServer does know that it is in transaction, because if I add a name to transaction statement it balks in rollback that it coul开发者_如何学Pythond not rollback transaction with that name. This gives me a clue that the TOP-LEVEL transaction is not started in PROC_1, but is indeed starts in JDBC.


Found a way.

Turns out when you calling

con.setAutoCommit(false)

is equivalent to setting IMPLICIT_TRANSACTIONS option on the connection, like this

SET IMPLICIT_TRANSACTIONS ON;

Then you can check implicit transaction status with the help of @@OPTIONS function:

if ((@@OPTIONS & 2) != 2) and (@@TRANCOUNT = 0)
begin
    select 'No transaction'
end
else begin
    select 'Already in transaction'
end
0

精彩评论

暂无评论...
验证码 换一张
取 消