我有一个线程,即通过tcp发送存储在类型为List 的缓冲区中的数据。另一个线程正在写入缓冲区。因为我对C#不太熟悉,所以我想知道如何正确使用lock或Mutex。

这是我最终想要使用的代码:

 while(buffer.isLocked())
 {
    buffer.wait();
 }

  buffer.lockBuffer();
  buffer.add(tcpPacket);
  buffer.unlockBuffer();
  buffer.notify();

这是我当前的代码。我希望有人能帮助我完成它。
public class Buffer
{
    private Mutex mutex;
    private List<string> buffer;
    private bool locked = false;

    public Buffer()
    {
        mutex = new Mutex(false);
        buffer = new List<string>();
    }

    public bool isLocked()
    {
        return locked;
    }

    public void lockBuffer()
    {
        if (!locked)
        {
            //...
           locked = true;
        }
    }

    public void unlockBuffer()
    {
        if(locked)
        {
            mutex.ReleaseMutex();
            locked = false;
        }
    }

    public void wait()
    {
        mutex.WaitOne();
    }

    public void notify()
    {
        //...
    }
}

最佳答案

如果使用System.Collections.Concurrent.BlockingCollection会更好。它不需要外部同步。

对于那些不使用4.0的人

using System;
using System.Collections.Generic;
using System.Threading;

namespace MyCollections
{
    public class BlockingQueue<T> : IDisposable
    {
        Queue<T> _Queue = new Queue<T>();
        SemaphoreSlim _ItemsInQueue = null;
        SemaphoreSlim _FreeSlots = null;
        int _MaxItems = -1;

        public BlockingQueue(int maxItems=Int32.MaxValue)
        {
            _MaxItems = maxItems;
            _ItemsInQueue = new SemaphoreSlim(0, maxItems);
            _FreeSlots = new SemaphoreSlim(maxItems, maxItems);
        }

        public void Dispose()
        {
            if (_ItemsInQueue != null) _ItemsInQueue.Dispose();
            if (_FreeSlots != null) _FreeSlots.Dispose();
        }

        public int Count
        {
            get { return _ItemsInQueue.CurrentCount; }
        }


        public void Add(T item)
        {
            if(_MaxItems != Int32.MaxValue) _FreeSlots.Wait();
            lock (this)
            {
                _Queue.Enqueue(item);
                _ItemsInQueue.Release();
            }
        }


        public T Take()
        {
            T item = default(T);
            _ItemsInQueue.Wait();
            lock (this)
            {
                 item = _Queue.Dequeue();
                 if (_MaxItems != Int32.MaxValue)  _FreeSlots.Release();
            }
            return item;
        }
    }
}

关于c# - 如何使用互斥锁,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/7841390/

10-11 22:26
查看更多