一、前言

因为很多人一般进行一系列相关数据库操作都是在存储过程里面,而且在存储过程用锁的写法也是很简单的,在这篇文章主要介绍一下C#后台代码用锁进行一系列事务操作,我建立一个简单的winform程序,然后做一个事务:修改指定ID的用户信息,然后新增一名用户信息,操作成功就提交事务,程序异常和数据库执行不成功都必须回滚事务!

二、存储过程的事务

写得比较简单,我想大家都知道怎么使用了,不懂的可以问我。

三、C#代码的事务

1.封装一个事务类,里面有事务的创建、提交事务、回滚事务和销毁事务的方法

    public class TransactionDal : SqlHelper
    {
        public DbConnection dbconnection = null;
        public DbTransaction transaction = null;

        public void BeginTransaction()
        {
            dbconnection = SqlHelper.CreateConnection();
            dbconnection.Open();
            transaction = dbconnection.BeginTransaction();

        }

        public void CommitTransaction()
        {
            if (null != transaction)
            {
                transaction.Commit();
            }

        }

        public void RollbackTransaction()
        {
            if (null != transaction)
            {
                transaction.Rollback();
            }

        }

        public void DisposeTransaction()
        {
            if (dbconnection.State == ConnectionState.Open)
            {
                dbconnection.Close();
            }
            if (null != transaction)
            {
                transaction.Dispose();
            }

        }
    }

2.封装简单的数据层用到事务的方法

 public class SqlHelper
    {
        private static readonly string constr = ConfigurationManager.ConnectionStrings["strCon"].ConnectionString;

        /// <summary>
        /// 有锁的事务方法
        /// </summary>
        /// <param name="tran"></param>
        /// <param name="sql"></param>
        /// <param name="pms"></param>
        /// <returns></returns>
        public static int ExecuteNonQuery(IDbTransaction tran,string sql, params SqlParameter[] pms)
        {

                using (SqlCommand cmd = new SqlCommand(sql, (SqlConnection)tran.Connection, (SqlTransaction)tran))
                {
                    if (pms != null)
                    {
                        cmd.Parameters.AddRange(pms);
                    }

                    return cmd.ExecuteNonQuery();

                }

        }

        public static DataTable ExecuteDataTable(string sql, params SqlParameter[] pms)
        {
            SqlDataAdapter adapter = new SqlDataAdapter(sql, constr);
            if (pms != null)
            {
                adapter.SelectCommand.Parameters.AddRange(pms);
            }
            DataTable dt = new DataTable();
            adapter.Fill(dt);
            return dt;

        }

        protected static System.Data.Common.DbConnection CreateConnection()
        {
            SqlConnection con = new SqlConnection(constr);
            return con;
        }
    }

3.winform程序

3.1程序界面

3.2C#代码修改用户信息并新增用户信息用到了事务

   public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            dataGridView1.DataSource = GetUsersTable();
        }

        /// <summary>
        /// 给dataGridView绑定数据源
        /// </summary>
        /// <returns></returns>
        private DataTable GetUsersTable()
        {
            string sql = "select * from Users";
            DataTable dt = SqlHelper.ExecuteDataTable(sql, null);
            return dt;
        }

        /// <summary>
        /// 点击修改并新增按钮
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void button_Click(object sender, EventArgs e)
        {
            string strUserID = txtID.Text.Trim();
            string strUserName = txtUserName.Text.Trim();
            string strAge = txtAge.Text.Trim();
            string strAddress = txtAddress.Text.Trim();
            string strNewUserName = txtNewUserName.Text.Trim();
            string strNewAge = txtNewAge.Text.Trim();
            string strNewAddress = txtNewAddress.Text.Trim();
            if (strUserID != "" && strUserName != "" && strAge != "" && strAddress != "" && strNewUserName != "" && strNewAge != "" && strNewAddress != "")
            {
                int error = UpdateUserMsg(strUserID, strUserName, strAge, strAddress, strNewUserName, strNewAge, strNewAddress);
                )
                {
                    dataGridView1.DataSource = GetUsersTable();//重新绑定学生数据源
                     MessageBox.Show("整个事务操作成功");
                }
                else
                {
                    MessageBox.Show("整个事务操作失败");
                }
            }
            else
            {
                MessageBox.Show("请填写完整信息");
            }
        }

        /// <summary>
        /// 修改指定学生信息
        /// </summary>
        /// <returns></returns>
        private int UpdateUserMsg(string strUserID, string strUserName, string strAge, string strAddress, string strNewUserName, string strNewAge, string strNewAddress)
        {
            TransactionDal dalTran = new TransactionDal(); //实例化封装好事务类的TransactionDal
            ;
            try
            {
                dalTran.BeginTransaction();  //这里开启事务锁

                string sql = string.Format("update Users set UserName='{0}',Age={1},Address='{2}' where UserID={3}", strUserName, strAge, strAddress, strUserID);
                int mod = SqlHelper.ExecuteNonQuery(dalTran.transaction, sql, null);  //传参:事务锁,sql, null  ,执行修改操作
                )//执行成功
                {
                    int mod2 = AddUser(dalTran.transaction, strNewUserName, strNewAge, strNewAddress);//如果一系列操作是相关的也要传递锁过去
                    )
                    {
                        dalTran.CommitTransaction();   //执行提交
                    }
                    else
                    { //执行失败回滚
                        error += ;
                        dalTran.RollbackTransaction();
                    }
                }
                else     //执行失败回滚
                {
                    error += ;
                    dalTran.RollbackTransaction();
                    return error;
                }
            }
            catch (Exception)
            {                 //执行异常回滚
                error += ;
                dalTran.RollbackTransaction();

            }
            finally
            {
                dalTran.DisposeTransaction(); //释放锁,释放连接实例
            }

            return error;
        }

        //新增学生,在整个事务中也需要传递同个事务事例
        private int AddUser(IDbTransaction tran, string strNewUserName, string strNewAge, string strNewAddress)
        {
            string sql = string.Format("insert into Users values('{0}',{1},'{2}')", strNewUserName, strNewAge, strNewAddress);
            int mod = SqlHelper.ExecuteNonQuery(tran, sql, null);
            return mod;
        }

    }

四、总结

C#事务锁要try{}catch{}finally{},在一系列相关操作的时候开启了事务锁就只能在同一个数据库连接实例进行锁的操作, 代码都写得比较简单,大家会用C#代码事务,封装合适自己使用的就好了。

C#代码启用事务锁Transaction进行一系列提交回滚操作的更多相关文章

  1. EF Core利用Transaction对数据进行回滚保护

    What? 首先,说一下什么是EF Core中的Transaction Transaction允许以原子方式处理多个数据库操作,如果事务已提交,则所有操作都应用于数据库,如果事务回滚,则没有任何操作应 ...

  2. JDBC事务提交/回滚实例

    以下是使用事务教程中描述的提交和回滚的代码示例. 此示例代码是基于前面章节中完成的环境和数据库设置编写的. 复制并将以下示例代码保存到:CommitAndRollback.java 中,编译并运行如下 ...

  3. Spring事务只对运行时异常回滚

    我们在使用Spring时候一般都知道事务在遇到异常的时候会回滚,岂不知Spring的事务默认只有在发生运行时异常即:RunTimeException时才会发生事务,如果一个方法抛出Exception或 ...

  4. Spring学习笔记——Spring事务仅仅对执行时异常回滚

    我们在使用Spring时候一般都知道事务在遇到异常的时候会回滚.岂不知Spring的事务默认仅仅有在发生执行时异常即:RunTimeException时才会发生事务,假设一个方法抛出Exception ...

  5. Android学习笔记_9_SQLiteOpenHelper对象之数据库增删改查以及事务回滚操作

    一.SQLite数据库: 在Android平台上,集成了一个嵌入式关系型数据库—SQLite,SQLite3支持 NULL.INTEGER.REAL(浮点数字).TEXT(字符串文本)和BLOB(二进 ...

  6. SQL的事务回滚操作带案例分析

    SET XACT_ABORT on  BEGIN TRAN  INSERT INTO [PDA_Action] VALUES ('採购入库1')  INSERT INTO [PDA_Action] V ...

  7. [转] C# mysql 事务回滚

    什么是数据库事务 数据库事务是指作为单个逻辑工作单元执行的一系列操作. 设想网上购物的一次交易,其付款过程至少包括以下几步数据库操作: · 更新客户所购商品的库存信息 · 保存客户付款信息--可能包括 ...

  8. spring boot 使用 mybatis 开启事务回滚 的总结

    1.前言 以前没有使用mybatis,可以关闭自动提交,然后做sql操作,对操作进行catch捕获异常, 如果没有异常则commit 提交 ,有异常则 rollback 回滚,新增的数据则删除 ,修改 ...

  9. Oracle 事务 锁

    一. 事务 是一系列的数据库操作,是数据库应用的基本逻辑单位以及并发控制的基本单位.所谓的事务,它是一个操作序列,这些操作要么都执行,要么都不执行,它是一个不可分割的工作单位. 要将有组语句作为事务考 ...

随机推荐

  1. (四)CodeMirror - API

    内容相关 cm.getValue() cm.setValue() cm.getRange() editor.getRange({line:1},{line:2}) // 获取内容块字符 cm.repl ...

  2. Win 7 通过事件管理器查看计算机开机关机时间

    控制面板-管理工具-事件查看器 视图中开机来源:Kernel-General 事件ID:13 关机来源:Kernel-General 事件ID:12

  3. Windows环境变量修改

    做开发的时候都设置过Windows的环境变量,一直觉得实在是难用之极.网上搜索过解决方案,有个开源的小程序,试过,有问题,没有办法正常运行.所以自己写一个,权当练手. 下载地址:http://file ...

  4. 一个经试用效果非常不错的数据库连接池--JAVA

    前言: 虽说现在许多企业级的应用服务器均自己带有数据库连接池功能,就连 Tomcat 也支持了这种功能.然而在许多时候,我们还是要使用数据库连接池,如:访问数据库的 Java 桌面应用程序等.这个数据 ...

  5. POJ1671 动态规划

    POJ1671 问题重述: 本题求解一首N行诗可能的押韵结构的数目.所谓押韵结构,指的是指定的行数之间必须押韵.例如一首3行诗的押韵结构可以是aaa, aab, aba, baa, abc 5种(aa ...

  6. JS判断访问设备、客户端操作系统类型

    先给出一个实例:判断windows.linux.android 复制以下代码另存为html文件即可. <html> <head> <title>判断操作系统< ...

  7. 从类的继承看socketserver源码

    当我们拿到一份python源代码,我们要怎么去看呢? 下面我们以socketserver为例,看下面的一段代码: #!/usr/bin/env python # -*- coding: UTF-8 - ...

  8. CentOS下安装无线网卡驱动 (转)

    1. 确定自己的网卡和内核版本:lspci | grep Network  #根据输出的信息确定网卡的型号.uname -a             #确定内核版本 2. 配置yum使用RPMForg ...

  9. 【BBST 之伸展树 (Splay Tree)】

    最近“hiho一下”出了平衡树专题,这周的Splay一直出现RE,应该删除操作指针没处理好,还没找出原因. 不过其他操作运行正常,尝试用它写了一道之前用set做的平衡树的题http://codefor ...

  10. Yii2 分页类的扩展和listview引用

    Yii2 本身提供了不错分页选项供用户设置,但是实际项目中我们往往需要复杂一些的分页样式,例如下图所示的效果,上下翻页可用和不可用均用图标来替换.