Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 1055|回复: 0

C#:利用“事务+乐观锁+version”解决并发下的数据一致性问题

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-8-26 17:57:20 | 显示全部楼层 |阅读模式

    本文重点介绍通过事务控制,利用数据库的乐观锁和时间戳,来解决并发(非高并发)环境下的脏读、幻读、不可重复读等问题,同时也能解决超卖等现象,对开发企业管理系统的朋友提供一个思路,为更突出主题思路,文涉及到SqlSugar的一些代码已隐去。

    1. 数据库建表

    CREATE TABLE dbo.Test
    (
          tId        INT IDENTITY NOT NULL
        , tName      NVARCHAR (20) NOT NULL
        , tSalary    DECIMAL (8, 2) NULL
        , tTimeStamp TIMESTAMP
        , PRIMARY KEY (tId)
    )
    

    2. 创建类

        public partial class Test
        {
            [SugarColumn(IsPrimaryKey =true,IsIdentity =true)]
            public int tId { get; set; }
            public string tName { get; set; }
            public decimal? tSalary { get; set; }
            [SugarColumn(IsOnlyIgnoreInsert = true)]
            public byte[] tTimeStamp { get; set; }
        }
    

    3. 代码示例

    static async Task Main(string[] args)
    {
    	for (int i = 1; i <= 5; i++)
    	{
    		Task.Factory.StartNew(async (id) =>
    		{
    			await Test((int)id);
    		}, i);
    	}
    }
    
    static async Task Test(int threadID)
    {
    	var db = SqlSugar.DB;
    	for (int k = 1; k <= 50; k++)
    	{
    		string log = string.Empty;
    		log += $"第{threadID,2}线程,第{k}次";
    		//客户端从数据库获取数据
    		var firstRead = await db.Queryable<Test>().SingleAsync(x => x.tId == 2);
    		log += $"   name:{firstRead.tName,5} version:{BitConverter.ToString(firstRead.tTimeStamp).Replace(" - ", "")}";
    		//客户端修改数据需要时间
    		Thread.Sleep(10);
    		try
    		{
    			db.Ado.BeginTran();
    			log += "    事务开始";
    			//提交修改前数据进行验证
    			var secondRead = await db.Queryable<Test>().SingleAsync(x => x.tId == 2);
    			if (BitConverter.ToString(secondRead.tTimeStamp) != BitConverter.ToString(firstRead.tTimeStamp))
    			{
    				log += $"    不可重复读,version:{BitConverter.ToString(secondRead.tTimeStamp).Replace(" - ", "")}";
    				throw new Exception();
    			}
    			var data = new Test { tId = 2, tName = $"{threadID}-{k}" };
    			var result = await db.Updateable(data).Where(c => c.tTimeStamp == firstRead.tTimeStamp).IgnoreColumns(ignoreAllNullColumns: true).ExecuteCommandAsync();
    			db.Ado.CommitTran();
    			log += result > 0 ? $"    修改成功,当前name:{data.tName}" : "    修改失败,数据被其它线程修改";
    		}
    		catch (Exception)
    		{
    			db.Ado.RollbackTran();
    			log += "    事务回滚";
    		}
    		finally
    		{
    			Console.WriteLine(log);
    		};
    		Thread.Sleep(10);
    	}
    }
    

    运行结果

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-1-21 15:27 , Processed in 0.060313 second(s), 29 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表