zoukankan      html  css  js  c++  java
  • Why you shouldn't use Entity Framework with Transactions

    Links

    EntityFramework

    This is a .net ORM Mapper Framework from Microsoft to help you talking with your Database in an object oriented manner. Wikipedia

    Database Transaction

    A database transaction, by definition, must be atomic, consistent, isolated and durable. Database practitioners often refer to these properties of database transactions using the acronym ACID. Transactions in a database environment have two main purposes:

    1. To provide reliable units of work that allow correct recovery from failures and keep a database consistent even in cases of system failure, when execution stops (completely or partially) and many operations upon a database remain uncompleted, with unclear status.
    2. To provide isolation between programs accessing a database concurrently. If this isolation is not provided, the program's outcome are possibly erroneous. Wikipedia

    .NET Transactions

    A .NET Transaction can be used in different ways by different frameworks to support transactions. The .NET Transaction itself is not connected with the database by any means. MSDN

    .NET Transactions and the EntityFramework

    If you are using the Entity Framework during an opened TransactionScope, EntityFramework will open a new Transaction right with the next command that will be sent to the Database (CRUD Operation).

    Consider this code block:

    using (var transaction = new System.Transactions.TransactionScope())
    {
        // DBC = Database Command
    
        // create the database context
        var database = new DatabaseContext();
    
        // search for the user with id #1
        // DBC: BEGIN TRANSACTION
        // DBC: select * from [User] where Id = 1
        var userA = database.Users.Find(1);
        // DBC: select * from [User] where Id = 2
        var userB = database.Users.Find(2);
        userA.Name = "Admin";
    
        // DBC: update User set Name = "Admin" where Id = 1
        database.SaveChanges();
    
        userB.Age = 28;
        // DBC: update User set Age = 28 where Id = 2
        database.SaveChanges();
    
        // DBC: COMMIT TRANSACTION
        transaction.Complete();
    }
    

    https://gist.github.com/SeriousM/e6b30db2b21e7e602655#file-bad_example-cs

    The database.SaveChanges() call sends your changes to the database and executes them but they are not really persisted because you are in the database transaction scope. transaction.Complete() actually finishes the database transaction and your data is saved.

    That behavior is actually cool and very useful, right?

    NO. Absolutely not. full stop.

    Why not using .NET Transactions along with EntityFramework

    The default isolation mode is read committed and fits perfectly to 99% of your needs, eg. reading data. When you want to save the changes you made to the database (Create, Update, Delete), EntityFramework is smart enough to create a transaction without your notice behind the scenes to wrap the changes. You can be sure that everything will be saved or every change will be discarded (Atomicity).

    By using transactions in EntityFramework, you change that behavior and force everyCRUD operation during a transaction scope to be executed in serializable isolation mode which is the highest and most blocking type. No process will be able to access the tables you have touched (even reading from it) during your transaction. That can lead to Deadlocks pretty fast and you want to avoid them at all costs!

    That's how the code looks like without the explicit usage of transactions:

    // DBC = Database Command
    
    // create the database context
    var database = new DatabaseContext();
    
    // search for the user with id #1
    // DBC: select * from [User] where Id = 1
    var userA = database.Users.Find(1);
    // DBC: select * from [User] where Id = 2
    var userB = database.Users.Find(2);
    userA.Name = "Admin";
    userB.Age = 28;
    
    // DBC: BEGIN TRANSACTION
    // DBC: update User set Name = "Admin" where Id = 1
    // DBC: update User set Age = 28 where Id = 2
    // DBC: COMMIT TRANSACTION
    database.SaveChanges();
    

    https://gist.github.com/SeriousM/e6b30db2b21e7e602655#file-good_example-cs

    Rule of Thumb: Save only once per task and don't use transactions.

    Edit: thanks to @seimur - One thread should have access to one instance of DbContext which works best in web applications where every request acts as one thread. In windows applications every command or task should have one DbContext which is not shared. If you share the DbContext between threads you might run into issues like having reads sneaked into a foreign transaction.

  • 相关阅读:
    DNS 域名系统服务
    tomcat
    mysql+redis
    centos7搭建lnmp
    redis安装
    redis 高级应用
    ubuntu,安装、配置和美化(1)
    解决大于5.7版本mysql的分组报错Expression #1 of SELECT list is not in GROUP BY clause and contains nonaggregated column 'userinfo.
    交换机常用命令
    SSH爆破应急响应
  • 原文地址:https://www.cnblogs.com/dufu/p/3961883.html
Copyright © 2011-2022 走看看