繁体   English   中英

从WCF服务调用基于任务的异步单向回调方法

[英]Calling a task-based asynchronous one way callback method from WCF service

我有一个WCF服务,它使用回调协定来通知客户端,类似于此

public interface IClientCallback
{
    [OperationContract(IsOneWay = true)]
    void NotifySomething();
}

和调用它的服务代码与此类似

void NotifySomething()
{
    try
    {
        this.callback.NotifySomething();
    }
    catch (Exception ex)
    {
        // Log the exception and eat it
    }
}

请注意,根据设计,回调通知是可选的 ,即很高兴,但不是必需的。 这就是为什么将其标记为OneWay ,并且实现吃掉异常的原因。

由于某些误解,我们认为采用非阻塞性的“忘却”方法就足够了。 但是当然那不是真的,因此在某些情况下它会阻塞一段时间,这会引起问题,因为它是从内部线程同步块中调用的。 因此我们决定通过如下更改定义使其异步

public interface IClientCallback
{
    [OperationContract(IsOneWay = true)]
    Task NotifySomething();
}

我对客户端实现没有问题,我的问题是如何从服务中调用它。 这是我想做的

async void NotifySomething()
{
    try
    {
        await this.callback.NotifySomething();
    }
    catch (AggregateException ex)
    {
        // Unwrap, log the exception(s) and eat it
    }
    catch (Exception ex)
    {
        // Log the exception and eat it
    }
}

现在,既然每个人都说async void不是一个好习惯,那么可以在这里使用它吗? 我还有什么其他选择? 在WCF服务上下文中执行此操作的推荐方法是什么?

您编写的方式非常安全,因为它可以处理异常。 您也可以编写一个可重用的扩展方法来做到这一点,这样就无需重复它。

也许是这样的:

public static class Extensions
{
    public static void FireAndForget(this Task task)
    {
        task.ContinueWith(t => 
        {
            // log exceptions
            t.Exception.Handle((ex) =>
            {
                Console.WriteLine(ex.Message);
                return true;
            });

        }, TaskContinuationOptions.OnlyOnFaulted);
    }
}

public async Task FailingOperation()
{
    await Task.Delay(2000);

    throw new Exception("Error");
}   

void Main()
{
    FailingOperation().FireAndForget();

    Console.ReadLine();
}

暂无
暂无

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

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