0

我有一个 Web Api,它调用另一个 Web api 调用来获取一些信息。为了使应用程序更具弹性,我按照以下步骤实现了 HttpTransientErrorDetectionStrategy:https ://alexandrebrisebois.wordpress.com/2013/02/21/defining-an-http-transient-error-detection-strategy-for-休息电话/

之后,我使用如下代码调用另一个 Web 应用程序:

RetryPolicy _retryPolicy = new RetryPolicy<HttpTransientErrorDetectionStrategy>(
    new ExponentialBackoff(retryCount: 2, minBackoff: TimeSpan.FromSeconds(0), maxBackoff: TimeSpan.FromSeconds(10), deltaBackoff: TimeSpan.FromSeconds(2)));

var _httpClient = new HttpClient
{
    BaseAddress = new Uri("http://www.microsoft.com")
};

HttpResponseMessage response = _retryPolicy.ExecuteAsync(async () => await _httpClient.GetAsync($"", HttpCompletionOption.ResponseContentRead)).Result;

_httpClient.GetAsync 调用卡住了,我不知道为什么。如果我删除 _retryPolicy,直接使用 _httpClient.GetAsync,它会在几秒钟内返回。

我在控制台应用程序上有类似的代码,用于调用相同的 Web 应用程序,并且工作正常,所以这似乎特定于我在我的 Web API 中使用它的方式。这旨在成为 Azure 上的应用程序,但当我在本地调试时也会发生这种情况。有人知道为什么会卡住吗?我该如何调试呢?

谢谢!

4

1 回答 1

1

我在控制台应用程序上有类似的代码,用于调用相同的 Web 应用程序,并且工作正常,所以这似乎特定于我在我的 Web API 中使用它的方式。

您发布的代码在这里被阻止:

HttpResponseMessage response = _retryPolicy.ExecuteAsync(...).Result;

不要阻塞异步代码。相反,使用await

HttpResponseMessage response = await _retryPolicy.ExecuteAsync(...);

如果我删除 _retryPolicy,直接使用 _httpClient.GetAsync,它会在几秒钟内返回。

如果您的原始代码被阻塞,并且您必须阻塞异步代码(出于某种原因),那么您可以使用ConfigureAwait(false)hack

HttpResponseMessage response = _retryPolicy.ExecuteAsync(async () => await _httpClient.GetAsync($"", HttpCompletionOption.ResponseContentRead).ConfigureAwait(false)).Result;

省略async/await

HttpResponseMessage response = _retryPolicy.ExecuteAsync(() => _httpClient.GetAsync($"", HttpCompletionOption.ResponseContentRead)).Result;

PS退房DecorrelatedJitterBackoffV2

于 2020-05-21T16:23:15.587 回答