Use of TransactionScope with read uncommitted - is with (nolock) in SQL necessary?

前端 未结 2 1940
旧巷少年郎
旧巷少年郎 2021-02-20 11:52

I am using FluentNHibernate, and I have a list of records, mapped to an SQL Server 2008 view. Dirty reads are OK with me, not locking the tables is a priority.

The SQL Q

相关标签:
2条回答
  • 2021-02-20 12:16

    Short answer: No

    Long answer:

    Just defining the TransactionScope does not define that any read or write will be invoked within a transaction.

    To run something within a transaction, you still have to open and commit a transaction!

    The TransactionOptions of the TransactionScope for Timeout and IsolationLevel just define the defaults for any transaction created within the scope without those options explicitly set. Actually the TransactionScope does create a Transaction but it will not be active without opening a new Transaction. Internally this will do some complex stuff, cloning the transaction etc... so lets ignore this...

    Without a transaction you cannot define the isolation level, any select statement will be run with IsolationLevel.ReadCommitted because this is the default of SQL Server.

    You can also query session.Transaction.IsActive to see if a transaction is currently active for the session!

    Lets have a look at the following code, I put some comments to make it a little bit more clear

    using (var scope = new TransactionScope(TransactionScopeOption.Required,
                        new TransactionOptions()
                        {
                            IsolationLevel = IsolationLevel.ReadUncommitted
                        }))
    {
    
        using (var session = sessionFactory.OpenSession())
        {
            // outside any transaction...
            var x = session.Transaction.IsActive; // false;
    
            // read will be done with SQL Server default (ReadCommited)
            var pp = session.Query<Page>().Where(p => p.Photos.Count() > 1).ToList();
    
            using (var transaction = session.BeginTransaction())
            {
                // will use ReadUncommitted according to the scope
                var y = session.Transaction.IsActive; // true;
    
                var p1 = session.Get<Page>(1);
    
                transaction.Commit();
            }
            using (var transaction = session.BeginTransaction(System.Data.IsolationLevel.ReadCommitted))
            {
                // will use ReadCommitted according to the transaction initialization
                var y = session.Transaction.IsActive; // true;
    
                var p1 = session.Get<Page>(1);
    
                transaction.Commit();
            }
    
            scope.Complete();
        }
    }
    

    You can also watch how SQL Server reacts to those settings by using the SQL Server Profiler.

    Just create a new Trace and watch out for the Audit Login event, the text of the event will include the isolation level and you can see that it actually does a Audit Login each time a transaction is created, for example

     set transaction isolation level read uncommitted
    

    --

    Please correct me if any of this information might be wrong, I just figured this out by myself so there might be some potential of failure ;)

    0 讨论(0)
  • 2021-02-20 12:35

    I'm not sure why, but the TransactionScope code did not work for me.

    I used this code. Starting a new session below makes sure that you don't have an existing transaction already. (There are other ways to do it of course)

            using (var session = _session.SessionFactory.OpenSession())
            using (var txn = session.BeginTransaction(IsolationLevel.ReadUncommitted))
            {
                // bunch of select statements.
            }
    

    In the NHibernate Profiler I could see this:

    begin transaction with isolation level ReadUncommitted
    
    0 讨论(0)
提交回复
热议问题