วิธีเรียกวิธีใด ๆ แบบอะซิงโครนัสใน c #


110

ใครช่วยช่วยแสดงตัวอย่างโค้ดเล็ก ๆ ที่สาธิตวิธีเรียกเมธอดแบบอะซิงโครนัสใน c # ได้ไหม

คำตอบ:


131

หากคุณใช้ actionBeginInvoke () คุณต้องเรียก EndInvoke ที่อื่น - มิฉะนั้นเฟรมเวิร์กจะต้องเก็บผลลัพธ์ของการเรียก async บนฮีปซึ่งส่งผลให้หน่วยความจำรั่วไหล

หากคุณไม่ต้องการข้ามไปที่ C # 5 ด้วยคีย์เวิร์ด async / await คุณสามารถใช้ไลบรารี Task Parallels ใน. Net 4 ได้ดีกว่าการใช้ BeginInvoke / EndInvoke มากและให้วิธีการยิงที่ชัดเจน - และลืมงาน async:

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 async / await เป็นเพียงแค่น้ำตาลในการสังเคราะห์รอบ ๆ ไลบรารีงาน


2
หากยังไม่ชัดเจนการตั้งค่าสุดท้าย re: async / await นั้นถูกต้อง แต่จะเปลี่ยนรูปลักษณ์ของโค้ดของคุณไปอย่างมาก
Gusdor

ฉันกำลังลองโดยใช้วิธีการที่สร้างเหตุการณ์จากนั้นจึงมอบหมายงานนี้ถูกต้องหรือไม่? ถ้าเป็นเช่นนั้นฉันจะยุติงานได้อย่างไร ไชโย
Joster


24

นี่คือวิธีการ:

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


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

แน่นอนว่าคุณต้องแทนที่Actionด้วยผู้รับมอบสิทธิ์ประเภทอื่นหากวิธีการนั้นมีลายเซ็นอื่น


1
เมื่อเราเรียก foo แล้วฉันจะส่งอาร์กิวเมนต์ที่คุณไม่ได้แสดงได้อย่างไร
Thomas

แทนค่าว่างคุณสามารถใส่วัตถุ ให้ Foo ใช้พารามิเตอร์อินพุตหนึ่งตัวของวัตถุประเภท จากนั้นคุณจะต้องโยนวัตถุให้เป็นประเภทที่เหมาะสมใน Foo
Denise Skidmore

4

ดูบทความ MSDN การเขียนโปรแกรมแบบอะซิงโครนัสกับ Async และรอว่าคุณสามารถเล่นกับสิ่งใหม่ ๆ ได้หรือไม่ ถูกเพิ่มไปยัง. 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));
        }
    }
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.