线程同步中异常情况的处理

简介:

 本文主要用来说明多线程中异常情况的处理。

      问题出现:使用Lock进行多线程中的同步的时候,如果在Lock块里面出现了异常,那么同步的资源(变量)就没有办法被释放,最终将导致线程死锁。

复制代码
 1 using System;
 2 using System.Collections.Generic;
 3 using System.Linq;
 4 using System.Text;
 5 using System.Threading;
 6 
 7 namespace testLockList
 8 {
 9     class Program
10     {
11 
12         const int ThreadMaxNum = 5;//线程的最大数目
13         static void Main(string[] args)
14         {
15             TestList myList = new TestList();
16             for (int i = 0; i < ThreadMaxNum; i++)//同时建立5个进程,调用Add方法,看锁住了List,是否能对List里面的单个元素进行修改
17             {
18                 Thread myth = new Thread(TestList.add);
19                 myth.Name = string.Format("线程{0}", i);
20                 myth.IsBackground = true;
21                 myth.Start();
22             }
23             while (TestList.AddInvokeCount != 5)
24             {
25                 Thread.Sleep(500);
26             }
27 
28         }
29     }
30 
31     public class TestList
32     {
33         static List<tester> myList = new List<tester>();//一个List
34         static Mutex myMutex = new Mutex();          //信号量
35         public static int AddInvokeCount = 0;        //用于记录Add被调用的次数
36         public TestList()
37         {
38             myList.Add(new tester(0, 0));
39         }
40         static public void add()
41         {
42 
43             lock (myList)
44             {
45 
46 
47                 myMutex.WaitOne();  //信号量P操作
48                 int Num = myList.Count + 1;
49                 Thread.Sleep(1000);//保证线程等待
50 
51                 myList.Add(new tester(Num, Num));
52                 Console.WriteLine(Num);
53                 Console.WriteLine("线程{0}:增加", Thread.CurrentThread.Name);
54                 if (AddInvokeCount == 3)     //这里故意引发一个Exception
55                 {
56                     int b = 0;
57                     int a = 1 / b;
58                 }
59                 AddInvokeCount++;
60             }
61 
62         }
63     }
64     /// <summary>
65     /// 这个类只是用于测试
66     /// </summary>
67     public class tester
68     {
69         int a;
70         int b;
71         public tester(int a, int b)
72         {
73             this.a = a;
74             this.b = b;
75         }
76 
77     }
78 }
复制代码

出现的问题截图,如图1所示。

图1 Lock块内出现异常导致进程死锁

解决方法:使用信号量加try_catch_finally处理异常情况

     注:以下代码跟上面的问题代码不同的只是add()函数

复制代码
 1         static public void add()
 2         {
 3 
 4             try
 5             {
 6                 myMutex.WaitOne();  //信号量P操作
 7                 int Num = myList.Count + 1;
 8                 Thread.Sleep(1000);//保证线程等待
 9 
10                 myList.Add(new tester(Num, Num));
11                 Console.WriteLine(Num);
12                 Console.WriteLine("线程{0}:增加", Thread.CurrentThread.Name);
13                 if (AddInvokeCount == 3)     //这里故意引发一个Exception
14                 {
15                     int b = 0;
16                     int a = 1 / b;  
17                 }
18                 AddInvokeCount++;
19 
20             }
21             catch (System.Exception ex)
22             {
23 
24             }
25             finally
26             {
27                 myMutex.ReleaseMutex(); //信号量V操作
28             }
29 }
复制代码

运行效果,见图2:

图2 修复后效果

2013-01-02  00:33:53


作者:kissazi2 
出处:http://www.cnblogs.com/kissazi2/ 
本文版权归作者所有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

转载:http://www.cnblogs.com/kissazi2/archive/2013/01/02/2841847.html

目录
相关文章
|
7月前
多线程-同步代码块中的隐患及解决办法
多线程-同步代码块中的隐患及解决办法
37 0
|
3月前
|
缓存 算法 Java
多线程04 死锁,线程可见性
多线程04 死锁,线程可见性
22 0
|
3月前
|
算法
出现线程死锁缺陷一般有那些原因?该怎么解决?
出现线程死锁缺陷一般有那些原因?该怎么解决?
29 1
|
3月前
|
Go
并发陷阱:死锁、活锁和饥饿
并发陷阱:死锁、活锁和饥饿
26 0
|
6月前
|
Java
|
8月前
|
Java
Java多线程:捕获线程异常
你处理过多线程中的异常吗?如何捕获多线程中发生的异常?捕获子线程的异常与捕获当前线程的异常一样简单吗?
Java多线程: 如何捕获多线程中的异常
Java多线程: 如何捕获多线程中的异常
Java多线程: 如何捕获多线程中的异常
由于不当的执行顺序导致的死锁
由于不当的执行顺序导致的死锁
|
安全 前端开发 算法
多线程关于无锁的线程是否安全问题
多线程关于无锁的线程是否安全问题
160 0
多线程关于无锁的线程是否安全问题
什么是线程同步?又如何解决线程同步问题?
解决多个线程同时操作一个资源的问题
180 0