Як викликати будь-який метод асинхронно в #


110

Може хтось, будь ласка, покаже мені невеликий фрагмент коду, який демонструє, як викликати метод асинхронно в c #?

Відповіді:


131

Якщо ви використовуєте action.BeginInvoke (), вам потрібно кудись зателефонувати EndInvoke - інакше фреймворк повинен утримувати результат виклику асинхронізації в купі, що призводить до витоку пам'яті.

Якщо ви не хочете переходити до C # 5 за допомогою ключових слів асинхрон / очікувати, ви можете просто скористатися бібліотекою завдань Паралелі в. і забудьте про завдання асинхронізації:

using System.Threading.Tasks;
...
void Foo(){}
...
new Task(Foo).Start();

Якщо у вас є методи для виклику, які приймають параметри, ви можете використовувати лямбда для спрощення виклику без необхідності створювати делегати:

void Foo2(int x, string y)
{
    return;
}
...
new Task(() => { Foo2(42, "life, the universe, and everything");}).Start();

Я майже впевнений (але, правда, не позитивно), що синтаксис асинхронізації / очікування C # 5 - це лише синтаксичний цукор навколо бібліотеки завдань.


2
Якщо це ще не було зрозуміло, остаточне припущення re: async / await є правильним, але це кардинально змінить те, як виглядає ваш код.
Гусдор

Я намагаюся це за допомогою методу, який створює подію, а потім делегує, чи правильно це? Якщо так, то як я можу закінчити завдання. Ура
Йостер


24

Ось спосіб це зробити:

// The method to call
void Foo()
{
}


Action action = Foo;
action.BeginInvoke(ar => action.EndInvoke(ar), null);

Звичайно, вам потрібно замінити Actionделегатом іншого типу, якщо метод має іншу підпис


1
коли ми називаємо foo, то як я можу передати аргумент, який u не показав?
Томас

На зміну нулю можна поставити об’єкт. Нехай Foo приймає один вхідний параметр об'єкта типу. Потім вам доведеться передати об'єкт відповідному типу в Foo.
Деніз Скідмор

4

Перегляньте статтю MSDN Асинхронне програмування за допомогою Async та Await, якщо ви можете дозволити собі грати з новими речами. Додано до .NET 4.5.

Приклад фрагмента коду за посиланням (який є самим із цього проекту зразкового коду MSDN ):

// Three things to note in the signature: 
//  - The method has an async modifier.  
//  - The return type is Task or Task<T>. (See "Return Types" section.)
//    Here, it is Task<int> because the return statement returns an integer. 
//  - The method name ends in "Async."
async Task<int> AccessTheWebAsync()
{ 
    // You need to add a reference to System.Net.Http to declare client.
    HttpClient client = new HttpClient();

    // GetStringAsync returns a Task<string>. That means that when you await the 
    // task you'll get a string (urlContents).
    Task<string> getStringTask = client.GetStringAsync("http://msdn.microsoft.com");

    // You can do work here that doesn't rely on the string from GetStringAsync.
    DoIndependentWork();

    // The await operator suspends AccessTheWebAsync. 
    //  - AccessTheWebAsync can't continue until getStringTask is complete. 
    //  - Meanwhile, control returns to the caller of AccessTheWebAsync. 
    //  - Control resumes here when getStringTask is complete.  
    //  - The await operator then retrieves the string result from getStringTask. 
    string urlContents = await getStringTask;

    // The return statement specifies an integer result. 
    // Any methods that are awaiting AccessTheWebAsync retrieve the length value. 
    return urlContents.Length;
}

Цитування:

Якщо AccessTheWebAsyncміж викликом GetStringAsync і очікуванням його завершення немає жодної роботи, ви можете спростити свій код, зателефонувавши і чекаючи в наступному одному виписці.

string urlContents = await client.GetStringAsync();

Більше деталей - за посиланням .


Як я можу використовувати цю техніку та встановити тайм-аут?
Su Llewellyn

1
public partial class MainForm : Form
{
    Image img;
    private void button1_Click(object sender, EventArgs e)
    {
        LoadImageAsynchronously("http://media1.santabanta.com/full5/Indian%20%20Celebrities(F)/Jacqueline%20Fernandez/jacqueline-fernandez-18a.jpg");
    }

    private void LoadImageAsynchronously(string url)
    {
        /*
        This is a classic example of how make a synchronous code snippet work asynchronously.
        A class implements a method synchronously like the WebClient's DownloadData(…) function for example
            (1) First wrap the method call in an Anonymous delegate.
            (2) Use BeginInvoke(…) and send the wrapped anonymous delegate object as the last parameter along with a callback function name as the first parameter.
            (3) In the callback method retrieve the ar's AsyncState as a Type (typecast) of the anonymous delegate. Along with this object comes EndInvoke(…) as free Gift
            (4) Use EndInvoke(…) to retrieve the synchronous call’s return value in our case it will be the WebClient's DownloadData(…)’s return value.
        */
        try
        {
            Func<Image> load_image_Async = delegate()
            {
                WebClient wc = new WebClient();
                Bitmap bmpLocal = new Bitmap(new MemoryStream(wc.DownloadData(url)));
                wc.Dispose();
                return bmpLocal;
            };

            Action<IAsyncResult> load_Image_call_back = delegate(IAsyncResult ar)
            {
                Func<Image> ss = (Func<Image>)ar.AsyncState;
                Bitmap myBmp = (Bitmap)ss.EndInvoke(ar);

                if (img != null) img.Dispose();
                if (myBmp != null)
                    img = myBmp;
                Invalidate();
                //timer.Enabled = true;
            };
            //load_image_Async.BeginInvoke(callback_load_Image, load_image_Async);             
            load_image_Async.BeginInvoke(new AsyncCallback(load_Image_call_back), load_image_Async);             
        }
        catch (Exception ex)
        {

        }
    }
    protected override void OnPaint(PaintEventArgs e)
    {
        if (img != null)
        {
            Graphics grfx = e.Graphics;
            grfx.DrawImage(img,new Point(0,0));
        }
    }
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.