Как использовать HttpWebRequest (.NET) асинхронно? - PullRequest
147 голосов
/ 14 октября 2008

Как я могу использовать HttpWebRequest (.NET, C #) асинхронно?

Ответы [ 8 ]

121 голосов
/ 15 октября 2008

Использование HttpWebRequest.BeginGetResponse()

HttpWebRequest webRequest;

void StartWebRequest()
{
    webRequest.BeginGetResponse(new AsyncCallback(FinishWebRequest), null);
}

void FinishWebRequest(IAsyncResult result)
{
    webRequest.EndGetResponse(result);
}

Функция обратного вызова вызывается, когда асинхронная операция завершена. Вы должны хотя бы позвонить EndGetResponse() из этой функции.

64 голосов
/ 14 декабря 2010

Учитывая ответ:

HttpWebRequest webRequest;

void StartWebRequest()
{
    webRequest.BeginGetResponse(new AsyncCallback(FinishWebRequest), null);
}

void FinishWebRequest(IAsyncResult result)
{
    webRequest.EndGetResponse(result);
}

Вы можете отправить указатель запроса или любой другой объект, подобный этому:

void StartWebRequest()
{
    HttpWebRequest webRequest = ...;
    webRequest.BeginGetResponse(new AsyncCallback(FinishWebRequest), webRequest);
}

void FinishWebRequest(IAsyncResult result)
{
    HttpWebResponse response = (result.AsyncState as HttpWebRequest).EndGetResponse(result) as HttpWebResponse;
}

Привет

59 голосов
/ 20 декабря 2012

Все до сих пор были неправы, потому что BeginGetResponse() выполняет некоторую работу с текущим потоком. Из документации :

Метод BeginGetResponse требует некоторых задач синхронной установки для завершено (разрешение DNS, обнаружение прокси и подключение через сокет TCP, например) до того, как этот метод станет асинхронным. В следствии, этот метод никогда не должен вызываться в потоке пользовательского интерфейса (UI) потому что это может занять значительное время (до нескольких минут в зависимости от настроек сети) для завершения начального синхронного Задачи установки до того, как будет сгенерировано исключение для ошибки или метод преуспевает.

Итак, чтобы сделать это правильно:

void DoWithResponse(HttpWebRequest request, Action<HttpWebResponse> responseAction)
{
    Action wrapperAction = () =>
    {
        request.BeginGetResponse(new AsyncCallback((iar) =>
        {
            var response = (HttpWebResponse)((HttpWebRequest)iar.AsyncState).EndGetResponse(iar);
            responseAction(response);
        }), request);
    };
    wrapperAction.BeginInvoke(new AsyncCallback((iar) =>
    {
        var action = (Action)iar.AsyncState;
        action.EndInvoke(iar);
    }), wrapperAction);
}

Затем вы можете сделать то, что вам нужно, с ответом. Например:

HttpWebRequest request;
// init your request...then:
DoWithResponse(request, (response) => {
    var body = new StreamReader(response.GetResponseStream()).ReadToEnd();
    Console.Write(body);
});
55 голосов
/ 11 апреля 2014

Самый простой способ - использовать TaskFactory.FromAsync из TPL . Это буквально пара строк кода при использовании вместе с новыми async / await ключевыми словами:

var request = WebRequest.Create("http://www.stackoverflow.com");
var response = (HttpWebResponse) await Task.Factory
    .FromAsync<WebResponse>(request.BeginGetResponse,
                            request.EndGetResponse,
                            null);
Debug.Assert(response.StatusCode == HttpStatusCode.OK);

Если вы не можете использовать компилятор C # 5, то вышеперечисленное можно выполнить с помощью метода Task.ContinueWith :

Task.Factory.FromAsync<WebResponse>(request.BeginGetResponse,
                                    request.EndGetResponse,
                                    null)
    .ContinueWith(task =>
    {
        var response = (HttpWebResponse) task.Result;
        Debug.Assert(response.StatusCode == HttpStatusCode.OK);
    });
7 голосов
/ 06 марта 2013

Я закончил тем, что использовал BackgroundWorker, он определенно асинхронный в отличие от некоторых из вышеупомянутых решений, он обрабатывает возврат в поток GUI для вас, и его очень легко понять.

Также очень легко обрабатывать исключения, так как они заканчиваются в методе RunWorkerCompleted, но обязательно прочитайте это: Необработанные исключения в BackgroundWorker

Я использовал WebClient, но, очевидно, вы могли бы использовать HttpWebRequest.GetResponse, если хотите.

var worker = new BackgroundWorker();

worker.DoWork += (sender, args) => {
    args.Result = new WebClient().DownloadString(settings.test_url);
};

worker.RunWorkerCompleted += (sender, e) => {
    if (e.Error != null) {
        connectivityLabel.Text = "Error: " + e.Error.Message;
    } else {
        connectivityLabel.Text = "Connectivity OK";
        Log.d("result:" + e.Result);
    }
};

connectivityLabel.Text = "Testing Connectivity";
worker.RunWorkerAsync();
5 голосов
/ 19 октября 2017

.NET изменился, так как многие из этих ответов были опубликованы, и я хотел бы предоставить более актуальный ответ. Используйте асинхронный метод для запуска Task, который будет работать в фоновом потоке:

private async Task<String> MakeRequestAsync(String url)
{    
    String responseText = await Task.Run(() =>
    {
        try
        {
            HttpWebRequest request = WebRequest.Create(url) as HttpWebRequest;
            WebResponse response = request.GetResponse();            
            Stream responseStream = response.GetResponseStream();
            return new StreamReader(responseStream).ReadToEnd();            
        }
        catch (Exception e)
        {
            Console.WriteLine("Error: " + e.Message);
        }
        return null;
    });

    return responseText;
}

Чтобы использовать асинхронный метод:

String response = await MakeRequestAsync("http://example.com/");

Обновление:

Это решение не работает для приложений UWP, которые используют WebRequest.GetResponseAsync() вместо WebRequest.GetResponse(), и при необходимости не вызывает методы Dispose(). У @dragansr есть хорошее альтернативное решение, которое решает эти проблемы.

3 голосов
/ 12 марта 2018
public static async Task<byte[]> GetBytesAsync(string url) {
    var request = (HttpWebRequest)WebRequest.Create(url);
    using (var response = await request.GetResponseAsync())
    using (var content = new MemoryStream())
    using (var responseStream = response.GetResponseStream()) {
        await responseStream.CopyToAsync(content);
        return content.ToArray();
    }
}

public static async Task<string> GetStringAsync(string url) {
    var bytes = await GetBytesAsync(url);
    return Encoding.UTF8.GetString(bytes, 0, bytes.Length);
}
3 голосов
/ 08 октября 2012
public void GetResponseAsync (HttpWebRequest request, Action<HttpWebResponse> gotResponse)
    {
        if (request != null) { 
            request.BeginGetRequestStream ((r) => {
                try { // there's a try/catch here because execution path is different from invokation one, exception here may cause a crash
                    HttpWebResponse response = request.EndGetResponse (r);
                    if (gotResponse != null) 
                        gotResponse (response);
                } catch (Exception x) {
                    Console.WriteLine ("Unable to get response for '" + request.RequestUri + "' Err: " + x);
                }
            }, null);
        } 
    }
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...