httpclient呼び出しからコンテンツ本文を取得する方法は?


108

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 =現在のメソッドのコードが最適化されているため式を評価できません。、Status =現在のメソッドのコードが最適化されているため式を評価できません。、Method =現在のメソッドのコードが最適化されているため式を評価できません。、結果=現在のメソッドのコードが最適化されているため、式を評価できません。

..実行を継続しますが、永久に実行されます。問題がわからない


_numRetriesはどこでどのように定義されていますか?
Nathan A

クラスのスコープ内にあり、コンストラクターで0で初期化されます。AnalyzeSingle()は、私が使用する唯一の場所です。
Sherman Szeto 2014年

デバッグモードで実行していますか?最適化された問題は、リリースモードで実行していることが原因である可能性があります。
Nathan A

私は現在Debug / iisExpressにいます
Sherman Szeto

回答:


176

await / asyncを使用する方法はせいぜい貧弱であり、従うのが難しくなります。あなたは、混合されているawaitTask'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);

ありがとう!! 私は過去数時間(MSDN + stackoverflow)async / awaitがどのように機能するかを取得しようと努めてきましたが、明らかにまだ完全に把握していません。提案するリソースはありますか?
Sherman Szeto 2014年

1
それをいじり続けるだけで、やがてコツがつかれるようになります。一挙に把握するのは大きなコンセプトです。
ネイサンA

まだ問題があります。元の投稿で問題を更新しました。問題は私が同期実行のためにコーディングしていることかもしれませんが、その問題を解決する方法がわかりません
Sherman Szeto 2014年

1
HttpClientはIDisposableを実装しているため、usingステートメントでラップすることをお勧めします。
Payam

2
@Payam実装しているIDisposableことは事実ですが、usingステートメントでラップしないでください。これはまれな例外です。詳細については、この投稿を参照してください:aspnetmonsters.com/2016/08/2016-08-27-httpclientwrong
maxshuty

62

使用したくない場合は、コードを強制的に同期して実行するように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;
 }  

2
@nbushnellがPostAsyncに.Resultを追加すると、非同期ではなくなります
Mike

6
@マイクはnbushnellが言っていることではありませんか?:-)
PoeHaH 2018

タイプはresponse何ですか?同様のコードがありますが、responseグローバルにする必要があるため、型が必要です。ありがとう。
Azurespot 2018

1
@AzureSpot:応答のタイプはHttpResponseMessageです。
RWC 2019
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.