C#线程同步深入

简介: C#线程同步问题深究及解决方案

先来一个线程同步问题;看如下代码

    {
        static int money = 10000;
  
        static void QuQian(string name)
        {
            Console.WriteLine(name + "查看一下余额" + money);
            int yue = money - 1;
            Console.WriteLine(name + "取钱");
            money = yue;//故意这样写,写成 money--其实就没问题
            Console.WriteLine(name+"取完了,剩"+money);  
        }

        static void Main(string[] args)
        {
            Thread t1 = new Thread(() =>
            {
                for (int i = 0; i < 1000; i++)
                {
                    QuQian("t1");
                }
            });
            Thread t2 = new Thread(() =>
            {
                for (int i = 0; i < 1000; i++)
                { QuQian("t2"); }
            });
            t1.Start(); t2.Start(); t1.Join(); t2.Join();
            Console.WriteLine("余额" + money);
            Console.ReadKey();
        }
    }

控制台运行,猜猜结果是多少呢?看下图运行结果
哈哈.png
细心的你肯定发现了,每次运行结果肯定都不一样,毕竟是线程同步嘛!

下面我们来说解决方案;

解决思路:使用同步的技术避免两个线程同时修改一个余额。

解决方法 1:最大粒度——同步方法。

QuQian 方法上标注 [MethodImpl(MethodImplOptions.Synchronized)],这样一个方法只能同时被 一个线程访问。

解决方法 2:对象互斥锁
        {
            lock (locker)
            {
                Console.WriteLine(name + "查看一下余额" + money);
                int yue = money - 1;
                Console.WriteLine(name + "取钱");
                money = yue;//故意这样写,写成 money--其实就没问题
                Console.WriteLine(name + "取完了,剩" + money);
            }
        } 

同一时刻只能有一个线程进入同一个对象的 lock 代码块。必须是同一个对象才能起到 互斥的作用。lock 后必须是引用类型,不一定是 object,只要是对象就行。 锁对象选择很重要,选不对起不到同步的作用;选不对可能会造成其他地方被锁,比如 用字符串做锁(因为字符串拘留池导致可能用的是其他地方也在用的锁) 两个方法如果都用一个对象做锁,那么访问A的时候就不能访问B,因此锁选择很重要。

解决方法 3(*):Monitor

其实 lock 关键字就是对 Monitor 的简化调用,lock 最终就编译成 Monitor,因此一般不 不直接用 Monitor 类:

       {
           Monitor.Enter(locker);//等待没有人锁定 locker 对象,我就锁定它,然后继续执行
           try  {
               Console.WriteLine(name + "查看一下余额" + money);
               int yue = money - 1;
               Console.WriteLine(name + "取钱");
               money = yue;//故意这样写,写成 money--其实就没问题
               Console.WriteLine(name + "取完了,剩" + money);
           }
           finally  {
               Monitor.Exit(locker);//释放 locker 对象的锁
               }
       }
       //Monitor 有 TryEnter 方法,如果 Enter 的时候有人在占用锁,它不会等待,而是会返回 false。
       static void F1(int i)
       {
           if (!Monitor.TryEnter(locker))
           {
               Console.WriteLine("有人在锁着呢");
               return;
           }
           Console.WriteLine(i);
           Monitor.Exit(locker);
       } 

           static void Main(string[] args)
           {
               Thread t1 = new Thread(() => { for (int i = 0; i < 10000; i++) { F1(i); } }); t1.Start();

               Thread t2 = new Thread(() => {
                   for (int i = 0; i < 10000; i++)
                   {
                        F1(i);
                   }
               }); t2.Start();

               Console.ReadKey();
           }

第三种解决方案了解即可。

相关文章
|
6月前
|
SQL 开发框架 安全
C#编程与多线程处理
【4月更文挑战第21天】探索C#多线程处理,提升程序性能与响应性。了解C#中的Thread、Task类及Async/Await关键字,掌握线程同步与安全,实践并发计算、网络服务及UI优化。跟随未来发展趋势,利用C#打造高效应用。
198 3
|
6月前
|
安全 编译器 C#
C#学习相关系列之多线程---lock线程锁的用法
C#学习相关系列之多线程---lock线程锁的用法
|
6月前
|
并行计算 安全 Java
C# .NET面试系列四:多线程
<h2>多线程 #### 1. 根据线程安全的相关知识,分析以下代码,当调用 test 方法时 i > 10 时是否会引起死锁? 并简要说明理由。 ```c# public void test(int i) { lock(this) { if (i > 10) { i--; test(i); } } } ``` 在给定的代码中,不会发生死锁。死锁通常是由于两个或多个线程互相等待对方释放锁而无法继续执行的情况。在这个代码中,只有一个线程持有锁,且没有其他线程参与,因此不
389 3
|
6月前
|
Java 调度 C#
C#学习系列相关之多线程(一)----常用多线程方法总结
C#学习系列相关之多线程(一)----常用多线程方法总结
|
6月前
|
C#
C#学习相关系列之多线程---ConfigureAwait的用法
C#学习相关系列之多线程---ConfigureAwait的用法
115 0
|
6月前
|
C#
C#学习相关系列之多线程---TaskCompletionSource用法(八)
C#学习相关系列之多线程---TaskCompletionSource用法(八)
171 0
|
6月前
|
Java C#
C#学习系列相关之多线程(五)----线程池ThreadPool用法
C#学习系列相关之多线程(五)----线程池ThreadPool用法
|
3月前
|
数据采集 XML JavaScript
C# 中 ScrapySharp 的多线程下载策略
C# 中 ScrapySharp 的多线程下载策略
|
2月前
|
安全 数据库连接 API
C#一分钟浅谈:多线程编程入门
在现代软件开发中,多线程编程对于提升程序响应性和执行效率至关重要。本文从基础概念入手,详细探讨了C#中的多线程技术,包括线程创建、管理及常见问题的解决策略,如线程安全、死锁和资源泄露等,并通过具体示例帮助读者理解和应用这些技巧,适合初学者快速掌握C#多线程编程。
78 0