繁体   English   中英

ASP.NET 核心 Web ZDB974238714CA8DE634A7CE1D08 应用程序是否可能出现死锁或应用程序挂起 state

[英]Is deadlock or application hung state possible for ASP.NET Core Web API application

我有一个 ASP.NET Core 3.1 Web API 应用程序,它有一个后台主机服务,我在其中编写了以下代码来模拟应用程序的死锁:

public class Worker : IHostedService
{
    public Task StartAsync(CancellationToken cancellationToken)
    {
        var a = new object();
        var b = new object();

        lock (a)
        {
            lock (b)
            {
            }
        }

        // other thread
        lock (b)
        {
            lock (a)
            {
            }
        }

        return Task.CompletedTask;
    }
}

该应用程序正在运行,但该应用程序不会死锁或挂起 state,我仍然可以点击 web api 端点,结果返回给我。

是否有可能使正在运行的 web 应用程序陷入死锁或挂起 state,以便 API 中的 NONE 没有给出任何 200 结果?

谢谢。

不确定 ASP.NET,根据您的代码示例,您可以尝试使用线程睡眠/延迟之类的东西,这样每个线程就不会太快地锁定资源。

可能是这样的:

        lock (a)
        {
            Thread.sleep(..duration..)

            lock (b)
            {
            }
        }

        // other thread
        lock (b)
        {
            Thread.sleep(..duration..)

            lock (a)
            {
            }
        }

正如所写,当前方法永远不会死锁。

锁定对象是该方法的本地对象。 正确的锁定必须使用 scope 之外的 object 方法。

在接下来的版本中,锁定对象是 static 到 class,锁定增加了一个微小的延迟来模拟正在执行的工作。 在这种情况下,应用程序将在方法的第二个入口处死锁。

public class Worker
{
    private static readonly object _a = new();
    private static readonly object _b = new();
    private const int _delay = 5;

    public Task StartAsync2( int n )
    {
        lock( _a )
        {
            Console.WriteLine( $"{n}: _a1 enter" );
            lock( _b )
            {
                Console.WriteLine( $"{n}: _b1 enter" );
                Thread.Sleep( _delay );
            }
            Console.WriteLine( $"{n}: _b1 exit" );
        }
        Console.WriteLine( $"{n}: _a1 exit" );

        // other thread
        lock( _b )
        {
            Console.WriteLine( $"{n}: _b2 enter" );
            lock( _a )
            {
                Console.WriteLine( $"{n}: _a2 enter" );
                Thread.Sleep( _delay );
            }
            Console.WriteLine( $"{n}: _a2 exit" );
        }
        Console.WriteLine( $"{n}: _b2 exit" );

        return Task.CompletedTask;
    }
}

以下是执行Worker导致死锁的方法:

var w = new Worker();
var t = new Task[5];

for( int i = 0; i < t.Length; i++ )
{
    int n = i;
    t[i] = Task.Run( () =>
    {
        Console.WriteLine( $"{n} start" );
        w.StartAsync2( n );
        Console.WriteLine( $"{n} end" );

        return Task.CompletedTask;
    } );
}

Task.WaitAll( t );

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM