4

できるだけ早くデータを返す必要がある Web サービスがあります。クエリできるソースが 2 つあります。Task.WhenAny()これらのソースを照会するために使用を返し、最初に返されるアイテムを返したいと思います。ローカル開発環境でコードを実行すると、コードは最速のタスクからデータを正常に返しますが、低速のタスクが戻ると、IIS Express ワーカー プロセスが null 参照例外でクラッシュします。

MyService.asmx

<%@ WebService Language="C#" CodeBehind="MyService.asmx.cs" Class="MyWebService.MyService" %>

MyService.asmx.cs

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.Services;

namespace MyWebService
{
    /// <summary>
    /// Summary description for Service1
    /// </summary>
    [WebService(Namespace = "http://tempuri.org/")]
    [WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)]
    [System.ComponentModel.ToolboxItem(false)]
    // To allow this Web Service to be called from script, using ASP.NET AJAX, uncomment the following line. 
    // [System.Web.Script.Services.ScriptService]
    public class MyService : System.Web.Services.WebService
    {
        private WebCore _webService;

        public MyService()
        {
            _webService = new WebCore();
        }


        [WebMethod]
        public int TestAsync()
        {
            return _webService.TestInt();
        }

    }
}

WebCore.cs

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Threading;
using System.Threading.Tasks;

namespace MyWebService
{
   public class WebCore {

       public WebCore() {}

        public int TestInt()
        {
            return AsyncCaller().Result;
        }
        private CancellationTokenSource cts;

        async Task<int> CallOne(CancellationToken ct)
        {
            await Task.Delay(4000);
            return 1;
        }
        async Task<int> CallTwo(CancellationToken ct)
        {
            await Task.Delay(1000);
            return 2;
        }
        private async Task<int> AsyncCaller()
        {
            cts = new CancellationTokenSource();
            var tasks = new List<Task<int>>()
            {
                CallOne(cts.Token),
                CallTwo(cts.Token)
            };

            var completedtasks = await Task.WhenAny(tasks);
            var res = await completedtasks;
            cts.Cancel();
            return res;
        }
   }
}

これをデバッグ モードで実行して Web サービスを実行すると、Web サービスは期待どおり "2" を返します。ただし、3 秒後に CallOne() が完了すると、「コール スタックには外部コードのみが含まれています」という null 参照例外「オブジェクト参照がオブジェクトのインスタンスに設定されていません」でアプリがクラッシュします。私の私のスタックトレースは次のとおりです。

 at System.Web.ThreadContext.AssociateWithCurrentThread(Boolean setImpersonationContext)
   at System.Web.HttpApplication.OnThreadEnterPrivate(Boolean setImpersonationContext)
   at System.Web.LegacyAspNetSynchronizationContext.CallCallbackPossiblyUnderLock(SendOrPostCallback callback, Object state)
   at System.Web.LegacyAspNetSynchronizationContext.CallCallback(SendOrPostCallback callback, Object state)
   at System.Web.LegacyAspNetSynchronizationContext.Post(SendOrPostCallback callback, Object state)
   at System.Threading.Tasks.SynchronizationContextAwaitTaskContinuation.PostAction(Object state)
   at System.Threading.Tasks.AwaitTaskContinuation.RunCallback(ContextCallback callback, Object state, Task& currentTask)
--- End of stack trace from previous location where exception was thrown ---
   at System.Threading.Tasks.AwaitTaskContinuation.<ThrowAsyncIfNecessary>b__1(Object s)
   at System.Threading.QueueUserWorkItemCallback.WaitCallback_Context(Object state)
   at System.Threading.ExecutionContext.RunInternal(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx)
   at System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx)
   at System.Threading.QueueUserWorkItemCallback.System.Threading.IThreadPoolWorkItem.ExecuteWorkItem()
   at System.Threading.ThreadPoolWorkQueue.Dispatch()
   at System.Threading._ThreadPoolWaitCallback.PerformWaitCallback()
4

1 に答える 1

6

まず、 ASP.NET でasync/を適切に有効にする必要があります。awaitこれを行うには、.NET 4.5 をターゲットにしていることを確認してから、 に設定httpRuntime.targetFrameworkするか、 ofセットを に4.5追加appSettingaspnet:UseTaskFriendlySynchronizationContexttrueします。

async次に、メソッドを完了するときにリクエスト コンテキストに再度入るのを避けたいと考えています。

async Task<int> CallOne(CancellationToken ct)
{
    await Task.Delay(4000).ConfigureAwait(false);
    return 1;
}
async Task<int> CallTwo(CancellationToken ct)
{
    await Task.Delay(1000).ConfigureAwait(false);
    return 2;
}
于 2013-09-04T20:52:23.057 に答える