Как получить тело содержимого из вызова httpclient?

111

Я пытался понять, как читать содержимое вызова httpclient, но не могу этого понять. Статус ответа, который я получаю, равен 200, но я не могу понять, как добраться до фактического возвращаемого Json, а это все, что мне нужно!

Вот мой код:

async Task<string> GetResponseString(string text)
{
    var httpClient = new HttpClient();

    var parameters = new Dictionary<string, string>();
    parameters["text"] = text;
    Task<HttpResponseMessage> response = 
        httpClient.PostAsync(BaseUri, new FormUrlEncodedContent(parameters));

    return await response.Result.Content.ReadAsStringAsync();
}

И я получаю его, просто вызывая его из метода:

Task<string> result =  GetResponseString(text);

И вот что я получаю

response Id = 89, Status = RanToCompletion, Method = "{null}", Result = "StatusCode: 200, ReasonPhrase: 'OK', Version: 1.1, Content: System.Net.Http.StreamContent, Headers:\r\n{\r\n Connection: keep-alive\r\n Date: Mon, 27 Oct 2014 21:56:43 GMT\r\n ETag: \"5a266b16b9dccea99d3e76bf8c1253e0\"\r\n Server: nginx/0.7.65\r\n Content-Length: 125\r\n Content-Type: application/json\r\n}" System.Threading.Tasks.Task<System.Net.Http.HttpResponseMessage>

Обновление: это мой текущий код на ответ Натана ниже

    async Task<string> GetResponseString(string text)
    {
        var httpClient = new HttpClient();

        var parameters = new Dictionary<string, string>();
        parameters["text"] = text;

        var response = await httpClient.PostAsync(BaseUri, new FormUrlEncodedContent(parameters));
        var contents = await response.Content.ReadAsStringAsync();

        return contents;
    }

И я вызываю это из этого метода ....

 string AnalyzeSingle(string text)
    {
        try
        {
            Task<string> result = GetResponseString(text);
            var model = JsonConvert.DeserializeObject<SentimentJsonModel>(result.Result);

            if (Convert.ToInt16(model.pos) == 1)
            {
                _numRetries = 0;
                return "positive";
            }

            if (Convert.ToInt16(model.neg) == 1)
            {
                _numRetries = 0;
                return "negative";
            }

            if (Convert.ToInt16(model.mid) == 1)
            {
                _numRetries = 0;
                return "neutral";
            }

            return "";
        }
        catch (Exception e)
        {
            if (_numRetries > 3)
            {
                LogThis(string.Format("Exception caught [{0}] .... skipping", e.Message));
                _numRetries = 0;
                return "";
            }
            _numRetries++;
            return AnalyzeSingle(text);
        }
    }

И он продолжает работать вечно, Он попадает в строку var model = JsonConvert.DeserializeObject<SentimentJsonModel>(result.Result); Один раз и продолжает работать, не останавливаясь на другой точке останова.

Когда я приостанавливаю исполнение, он говорит

Id = Невозможно оценить выражение, потому что код текущего метода оптимизирован., Статус = Невозможно оценить выражение, потому что код текущего метода оптимизирован., Метод = Невозможно оценить выражение, поскольку код текущего метода оптимизирован., Результат = Невозможно оценить выражение, потому что код текущего метода оптимизирован.

.. Продолжаю выполнение, но оно просто продолжается вечно. Не уверен, в чем проблема

Шерман Сзето
источник
Где и как определяется _numRetries?
Nathan A
Он находится в области действия класса и инициализируется 0 в конструкторе. AnalyzeSingle () - единственное место, где я его использую.
Шерман Сзето
Вы работаете в режиме отладки? Оптимизированная проблема может быть связана с тем, что вы работаете в режиме выпуска.
Nathan A
Я сейчас на Debug / iisExpress
Шерман

Ответы:

178

То, как вы используете await / async, в лучшем случае оставляет желать лучшего, и за ним трудно следить. Вы смешиваете awaitс Task'1.Result, что просто сбивает с толку. Однако похоже, что вы смотрите на конечный результат задачи, а не на ее содержимое.

Я переписал вашу функцию и вызов функции, что должно решить вашу проблему:

async Task<string> GetResponseString(string text)
{
    var httpClient = new HttpClient();

    var parameters = new Dictionary<string, string>();
    parameters["text"] = text;

    var response = await httpClient.PostAsync(BaseUri, new FormUrlEncodedContent(parameters));
    var contents = await response.Content.ReadAsStringAsync();

    return contents;
}

И ваш последний вызов функции:

Task<string> result = GetResponseString(text);
var finalResult = result.Result;

Или даже лучше:

var finalResult = await GetResponseString(text);
Натан А
источник
Спасибо!! Я пытался понять, как работает async / await последние пару часов (MSDN + stackoverflow), но я, очевидно, еще не понял это полностью. Можете ли вы предложить какие-либо ресурсы?
Шерман Сзето
1
Просто продолжайте играть с ним, и в конце концов вы научитесь этому. Это большая концепция, которую нужно усвоить сразу.
Nathan A
У меня все еще проблемы. Я обновил свою проблему в исходном сообщении. Проблема может заключаться в том, что я кодирую синхронное выполнение, но я не уверен, как решить эту проблему,
Шерман Сзето
1
HttpClient реализует IDisposable, поэтому лучше заключить его в оператор using.
Payam
2
@Payam, хотя это правда, что он реализует, IDisposableвы не должны заключать его в usingзаявление. Это редкое исключение из правил. См. Этот пост для получения дополнительной информации: aspnetmonsters.com/2016/08/2016-08-27-httpclientwrong
maxshuty
63

Если вы не хотите использовать, asyncвы можете добавить, .Resultчтобы код выполнялся синхронно:

private string GetResponseString(string text)
{
    var httpClient = new HttpClient();

    var parameters = new Dictionary<string, string>();
    parameters["text"] = text;

    var response = httpClient.PostAsync(BaseUri, new FormUrlEncodedContent(parameters)).Result;
    var contents = response.Content.ReadAsStringAsync().Result;

    return contents;
 }  
Nbushnell
источник
2
@nbushnell, добавив .Result в ваш PostAsync, делает его неасинхронным
Майк,
6
@ Майк, разве это не то, что говорит Бушнелл? :-)
PoeHaH
Для чего нужен тип response? У меня есть похожий код, но мне нужно сделать responseглобальным, поэтому мне нужен тип. Спасибо.
Azurespot
1
@AzureSpot: Тип ответа - HttpResponseMessage.
RWC