-4

我有一个方法“Add2List”,它创建一个 ManualResetEvent 并将其存储在实例的 SortedList 中,然后等待信号,然后做一些工作并处理事件。
我有另一种方法“DoSomething”,它监听远程服务器,然后根据 Guid 向存储的手动事件发出信号。

在多线程上下文中,多线程调用方法“Add2List”,因此在 sortedlist 中可能同时存在多个同名的手动事件。但这可能会导致混乱。我应该如何避免这种情况?

为了简单起见,我写了这个测试代码:

Class Program
{
  static void Main(string[] args)
  {
    StringBuilder str = new StringBuilder();//a string to record what happened
    //test iteratively
    for(int i=0;i<100;i++)
    {
      EventHolder holder = new EventHolder();
      Signaler ob2 = new Signaler();
      Thread th1 = new Thread(holder.Add2List);
      Thread th2 = new Thread(holder.Add2List);
      Thread th3 = new Thread(ob2.DoSomething);
      th1.Start(1);
      th2.Start(2);
      th3.Start();
      //Make sure all thread is ended before the next iteration.
      while(th1.IsAlive){ Thread.Sleep(200); }
      while(th2.IsAlive){ Thread.Sleep(200); }
      while(th3.IsAlive){ Thread.Sleep(200); }
    }
    Console.Read();
  }

  public class EventHolder
  {
    static SortedList<int, ManualResetEvent> MyManualEventList = new SortedList<int, ManualResetEvent>();
    public EventHolder()
    {
      MyManualEventList = new SortedList<int, ManualResetEvent>();
      Signaler.SignalMyManualEvent += OnSignalMyManualEvent;
    }
    void OnSignalMyManualEvent(int listindex)
    {
      try { MyManualEventList[listindex].Set(); }
      catch(Exception e)
      {
        Console.WriteLine("Exception throws at " + System.DateTime.Now.ToString() +" Exception Message:"
        Console.WriteLine(e.Message);
        int temp = 0; //*Here is a breakpoint! To watch local variables when exception happens.
      }
    }
    public void Add2List(object listindex)
    {
      ManualResetEvent MyManualEvent = new ManualResetEvent(false);
      MyManualEvent.Reset();
      MyManualEventList.Add((int)listindex, eve);
      //in this test, this countdownevent need to be signaled twice, for it has to wait until all 2 event been added to MyManualEventList
      Signaler.StartTrySignal.Signal();
      MyManualEvent.WaitOne();
      Console.WriteLine("Event" + ((int)listindex).ToString() + " been detected at " + System.DateTime.Now.Tostring());
      MyManualEvent.Dispose();
    }
  }

  public class Signaler
  {
    public delegate void Signalhandler(int listindex);
    public static event Signalhandler SignalMyManualEvent;
    public static CountDownEvent StartTrySignal = new CountDownEvent(2); // signaled twice so that the 2 manual events were added to sortedlist
    void RaiseSignalMyManualEvent(int listindex)
    {
      var vr = SignalMyManualEvent;
      if(vr != null)
        vr(listindex);
    }
    int i = 0, j = 0, k = 0;
    // here i use 2 prime numbers to simulate the happening of 2 random events
    public Signaler()
    {
      StartTrySignal.Reset();
    }
    public void DoSomething()
    {
      StartTrySignal.Wait(); // wait for the 2 manual events been added to sortedlist
      //To signal MyManualEventList[1] or MyManualEventList[2]
      while(i + j == 0)
      {
        Random rnd = new Random();
        k = rnd.Next();
        if(k % 613 == 0) { i = 1; Console.WriteLine("Event1 Raised!"; RaiseSignalMyManualEvent(1); }
        else if(k % 617 == 0) { j = 1;  Console.WriteLine("Event1 Raised!"; RaiseSignalMyManualEvent(2); }
      }
      //if MyManualEventList[1] has not been signaled, wait something to happen, and signal it.
      while(i == 0)
      {
        Random rnd = new Random();
        k = rnd.Next();
        if(k % 613 == 0)
        {
          i = 1;
          if(j>0)
          {
            m++;
            Console.WriteLine("All 2 Events Raised!  - iteration " + m.ToString());
          }
          RaiseSignalMyManualEvent(1);
        }
      }
      //if MyManualEventList[2] has not been signaled, wait something to happen, and signal it.
      while(j == 0)
      {
        Random rnd = new Random();
        k = rnd.Next();
        if(k % 617 == 0)
        {
          j = 1;
          m++;
          Console.WriteLine("All 2 Events Raised!  - iteration " + m.ToString());
          RaiseSignalMyManualEvent(2); 
        }
      }
    }
  }
  public class Counter //Provide a number to record iteration
  {
    public static int m = 0;
  }
}

结果:抱歉没有足够的声望来发布图片。
在有断点的那一行,系统抛出异常“给定的键不在字典中”。此异常是随机发生的,有时是因为 th1 已处置 <2, MyManualEvent> 或 th2 已处置 <1, MyManualEvent> ,有时没有已处置但它就是找不到任何人。我运行这个程序 3 次,异常发生在迭代 12、迭代 45 和迭代 0(在开始时)。

4

1 回答 1

1

好的 2 个回答

1:您的代码在“所有事件”之后返回“事件 1”,因为两个 console.writelines 处于竞争状态(最后一个 while 循环从不迭代)

2:“系统”区分两个 ManualResetEvent 对象,因为它引用了您放入它们的 SortedList。即。

static SortedList<int, ManualResetEvent> MyManualEventList 
       = new SortedList<int, ManualResetEvent>();

  public EventHolder() { Signaler.SignalMyManualEvent 
       += OnSignalMyManualEvent; }

  void OnSignalMyManualEvent(int listindex)
  {

MyManualEventList[ listindex ].Set();

  }

当您引发事件 1 时,您在 SortedList 中的第 1 项上调用 set,当您引发事件 2 时,您在列表中的第 2 项上调用 set。

这很糟糕,因为调用代码不知道它允许哪个线程继续,你很可能会得到一个空异常

于 2015-04-07T16:30:29.690 回答