123 votes

Comment appeler une méthode de manière asynchrone en c# ?

Pourrait-on me montrer un petit extrait de code qui démontre comment appeler une méthode de manière asynchrone en c# ?

141voto

Drew Shafer Points 2072

Si vous utilisez action.BeginInvoke(), vous doivent appeler EndInvoke quelque part - sinon le cadre doit conserver le résultat de l'appel asynchrone sur le tas, ce qui entraîne une fuite de mémoire.

Si vous ne voulez pas passer à C# 5 avec les mots-clés async/await, vous pouvez simplement utiliser la bibliothèque Task Parallels dans .Net 4. C'est beaucoup, beaucoup plus agréable que d'utiliser BeginInvoke/EndInvoke, et cela donne un moyen propre de déclencher et d'oublier les tâches asynchrones :

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

Si vous avez des méthodes à appeler qui prennent des paramètres, vous pouvez utiliser une lambda pour simplifier l'appel sans avoir à créer des délégués :

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

Je suis à peu près sûr (mais je ne suis pas certain) que la syntaxe async/await de C# 5 n'est que du sucre syntaxique autour de la bibliothèque Task.

2 votes

Si ce n'était pas déjà clair, l'hypothèse finale concernant async/await est correcte mais elle changera radicalement l'aspect de votre code.

0 votes

J'essaie de le faire avec une méthode qui crée un événement et qui délègue ensuite, est-ce correct ? Si oui, comment puis-je terminer la tâche ? Je vous remercie.

64voto

ms007 Points 1186

À partir de .Net 4.5, vous pouvez utiliser Task.Run pour lancer simplement une action :

void Foo(string args){}
...
Task.Run(() => Foo("bar"));

Task.Run vs Task.Factory.StartNew

26voto

Thomas Levesque Points 141081

Voici comment procéder :

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

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

Bien entendu, vous devez remplacer Action par un autre type de délégué si la méthode a une signature différente

1 votes

Lorsque nous appelons foo, comment puis-je passer l'argument que vous n'avez pas montré ?

0 votes

On peut mettre un objet à la place de null. Demandez à Foo de prendre un paramètre d'entrée de type objet. Vous devrez ensuite convertir l'objet dans le type approprié dans Foo.

5voto

Michael Blake Points 879

Consultez l'article de MSDN Programmation asynchrone avec Async et Await si vous pouvez vous permettre de jouer avec du nouveau matériel. Il a été ajouté à .NET 4.5.

Exemple d'extrait de code tiré du lien (qui est lui-même tiré de ce projet de code d'exemple 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;
}

Citation :

Si AccessTheWebAsync n'a pas de travail à faire entre l'appel à GetStringAsync et l'attente de son achèvement, vous pouvez simplifier votre code en appelant et en attendant dans la seule instruction suivante.

string urlContents = await client.GetStringAsync();

Plus de détails sont disponibles dans le lien .

0 votes

Comment puis-je utiliser cette technique et définir un délai d'attente ?

1voto

Dr.Sai Points 11
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));
        }
    }

Prograide.com

Prograide est une communauté de développeurs qui cherche à élargir la connaissance de la programmation au-delà de l'anglais.
Pour cela nous avons les plus grands doutes résolus en français et vous pouvez aussi poser vos propres questions ou résoudre celles des autres.

Powered by:

X