Monday, 8 April 2013

what is deadlock and How to handle Transaction Deadlocks?



  Firstly we should know about deadlock.

Deadlocks occur when two users have locks on separate objects and each user wants a lock on the other's object. When this happens, SQL Server ends the deadlock by automatically choosing one and aborting the process, allowing the other process to continue. The aborted transaction is rolled back and an error message is sent to the user of the aborted process. Generally, the transaction that requires the least amount of overhead to rollback is the transaction that is aborted.

  Let us use a scenario where a transaction A attempts to update table 1 and subsequently read/update data from table 2. At the same time there is another transaction B  which is trying to update table 2, and subsequently read /update data from table 1. In this scenario, transaction X holds a lock that transaction Y needs to complete its tasks and vice versa. So in this scenario neither transaction can complete until the other transaction is release.

Transaction deadlock situation:

Transaction A:

BEGIN TRAN

UPDATE EMPLOYEE SET EMPLOYEENAME=' XYZ' WHERE EMPLOYEEID=111
WAITFOR DELAY '00:00:05'
UPDATE SALARY SET BASIC= 200 WHERE EMPLOYEEID=111
COMMIT TRAN

Transaction B:

BEGIN TRAN
UPDATE SALARY SET HRA=200 WHERE EMPLOYEEID=111
WAITFOR DELAY '00:00:05'
UPDATE EMPLOYEE SET EMPLOYEENAME='ABC' WHERE EMPLOYEEID=111
COMMIT TRAN

Result:

(1 row(s) affected)
Msg 1205, Level 13, State 45, Line 5
Transaction (Process ID 53) was deadlocked on lock resources with another process and has been chosen as the deadlock victim. Rerun the transaction.

 Reason for above deadlock:

What we did is we copied these two transactions to two different query windows and run them simultaneously. Consequently what happened is that Transaction X locks and updates Employee table whereas transaction X locks and updates Salary table. After a delay of 20 ms, transaction X looks for the lock on Salary table which is already held by transaction Y and transaction Y looks for lock on Employee table which is held by transaction X. So both the transactions cannot proceed further; the deadlock occurs and the SQL server returns the error message 1205 for the aborted transaction.


How deadlock is resolved:

The user can choose which process should stop to allow another process to continue. SQL Server automatically chooses the process to terminate which is running completes the circular chain of locks. Sometime, it chooses the process running for a shorter period than another process. But it is recommended that we should provide a solution for handling deadlocks by finding the problem in our query code and then modify our processing to avoid deadlock situations.

Let us rewrite our transaction query.

Transaction A:

RETRY:
BEGIN TRAN
BEGIN TRY
      UPDATE EMPLOYEE SET EMPLOYEENAME='XYZ' WHERE EMPLOYEEID=111
      WAITFOR DELAY '00:00:10'
      UPDATE SALARY SET BASIC= 100 WHERE EMPLOYEEID=111
      COMMIT TRAN
END TRY
BEGIN CATCH
      PRINT 'Rollback Transaction'
      ROLLBACK TRANSACTION
      IF ERROR_NUMBER() =1205 -- DEADLOCK NUMBER
      BEGIN
            WAITFOR DELAY '00:00:00.05'
            GOTO RETRY
      ENDEND CATCH

Transaction B:

RETRY:
BEGIN TRAN
BEGIN TRY
      UPDATE SALARY SET HRA=200 WHERE EMPLOYEEID=111
      WAITFOR DELAY '00:00:10'
      UPDATE EMPLOYEE SET EMPLOYEENAME='ABC' WHERE EMPLOYEEID=111
      COMMIT TRAN
END TRY
BEGIN CATCH
      PRINT 'Rollback Transaction'
      ROLLBACK TRANSACTION
      IF ERROR_NUMBER()=1205
      BEGIN
            WAITFOR DELAY '00:00:00.05'
            GOTO RETRY
      END
END CATCH

Result:

If we run these two Trans statement at the same time, we will get the result below.

(1 row(s) affected)
Rollback Transaction

(1 row(s) affected)

(1 row(s) affected)

  In this way we can avoid deadlock in transaction.

Saturday, 6 April 2013

What is index hint?


  I can ask this question another way

  suppose I have one table and 20 indexes are  on the table. But i want to force on query to use  index which i think it will give fast result .
  So is it possible ?

Ans - Yes. Through index hint it is possible.
     
   I think you got answer.Index hint means  forcing  query to use specified index not the index selected by query optimiser.

e,g

create table Movie(name varchar(100),id int identity(1,1))

create nonclustered index Movie_Name_I on Movie(name)

select * from name  with (index(Movie_Name_I ,nolock)) where name='xyz'