69

我需要尝试锁定一个对象,如果它已经锁定就继续(超时后,或者没有它)。

C# lock 语句被阻塞。

4

5 回答 5

109

Ed 有适合您的功能。只是不要忘记打电话Monitor.Exit()。您应该使用try-finally块来保证正确清理。

if (Monitor.TryEnter(someObject))
{
    try
    {
        // use object
    }
    finally
    {
        Monitor.Exit(someObject);
    }
}
于 2008-08-12T06:37:34.053 回答
43

我相信你可以使用Monitor.TryEnter().

lock 语句只是转换为一个Monitor.Enter()调用和一个try catch块。

于 2008-08-12T06:29:24.070 回答
17

我有同样的问题,我最终创建了一个TryLock实现 IDisposable 的类,然后使用该using语句来控制锁的范围:

public class TryLock : IDisposable
{
    private object locked;

    public bool HasLock { get; private set; }

    public TryLock(object obj)
    {
        if (Monitor.TryEnter(obj))
        {
            HasLock = true;
            locked = obj;
        }
    }

    public void Dispose()
    {
        if (HasLock)
        {
            Monitor.Exit(locked);
            locked = null;
            HasLock = false;
        }
    }
}

然后使用以下语法进行锁定:

var obj = new object();

using (var tryLock = new TryLock(obj))
{
    if (tryLock.HasLock)
    {
        Console.WriteLine("Lock acquired..");
    }
}
于 2014-05-19T00:22:30.267 回答
4

考虑使用带有超时输入的 AutoResetEvent 及其方法 WaitOne。

static AutoResetEvent autoEvent = new AutoResetEvent(true);
if(autoEvent.WaitOne(0))
{
    //start critical section
    Console.WriteLine("no other thread here, do your job");
    Thread.Sleep(5000);
    //end critical section
    autoEvent.Set();
}
else
{
    Console.WriteLine("A thread working already at this time.");
}

请参阅https://msdn.microsoft.com/en-us/library/cc189907(v=vs.110).aspx https://msdn.microsoft.com/en-us/library/system.threading.autoresetevent(v =vs.110).aspxhttps://msdn.microsoft.com/en-us/library/cc190477(v=vs.110).aspx

于 2017-02-03T18:15:26.133 回答
3

既然其他人已经为您指明了正确的方向,您可能会自己发现这一点,但 TryEnter 也可以采用超时参数。

Jeff Richter 的“CLR Via C#”是一本关于 CLR 内部细节的优秀书籍,如果您正在研究更复杂的内容。

于 2008-08-12T07:54:39.607 回答