Sebab kaedah tak segerak C# tergantung apabila mengakses hasil Tugas
Apabila menggunakan kata kunci async
dan await
C# untuk pengaturcaraan tak segerak, binaan tertentu boleh menyebabkan tingkah laku yang tidak dijangka dan kemungkinan kebuntuan.
Pertimbangkan senario berikut: Aplikasi berbilang peringkat menggunakan kaedah utiliti pangkalan data lanjutan ExecuteAsync
yang melaksanakan pertanyaan SQL secara tak segerak dan mengembalikan hasilnya. Kaedah lapisan tengah GetTotalAsync
memanggil ExecuteAsync
untuk mendapatkan semula data dan menyimpan hasil dalam pembolehubah asyncTask
. Akhir sekali, operasi UI cuba mengakses hasil secara serentak menggunakan asyncTask.Result
. Walau bagaimanapun, aplikasi itu digantung selama-lamanya.
Punca kebuntuan
Masalah timbul daripada menggunakan GetTotalAsync
dalam kaedah await
. Secara lalai, kesinambungan kaedah async dihantar pada SynchronizationContext
yang sama yang memulakan kaedah tersebut. Dalam kes ini, apabila menggunakan await
pada urutan UI, kesinambungan (return result;
) juga dijadualkan untuk dijalankan pada urutan UI.
Apabila asyncTask.Result
dipanggil pada urutan UI, ia menyekat urutan apabila Tugasan selesai. Walau bagaimanapun, kesinambungan yang dijadualkan pada urutan UI tidak boleh dilaksanakan sehingga asyncTask.Result
selesai. Ini mewujudkan jalan buntu di mana tiada utas boleh meneruskan pelaksanaan.
Penyelesaian
Untuk menyelesaikan kebuntuan ini, terdapat beberapa kaedah:
1. Padamkan kata kunci Async:
Hapuskan penggunaan await
dan tulis semula kaedah ExecuteAsync
dan GetTotalAsync
sebagai kaedah tak segerak tulen yang tidak menunggu:
<code class="language-csharp">public static Task<T> ExecuteAsync<T>(this OurDBConn dataSource, Func<OurDBConn, T> function) { // ... (代码保持不变) } public static Task<ResultClass> GetTotalAsync(...) { // ... (代码保持不变) }</code>
2. Gunakan ConfigureAwait:
Gunakan ConfigureAwait(false)
untuk menyatakan bahawa kesinambungan tidak sepatutnya dijadualkan pada urutan UI:
<code class="language-csharp">public static async Task<ResultClass> GetTotalAsync(...) { var resultTask = this.DBConnection.ExecuteAsync<ResultClass>( ds => ds.Execute("select slow running data into result")); return await resultTask.ConfigureAwait(false); }</code>
Perhatikan bahawa pendekatan ini memerlukan spesifikasi eksplisit await
pada semua operasi ConfigureAwait(false)
yang mungkin membawa kepada kebuntuan.
3. Gunakan Konteks Penyegerakan:
Buat SynchronizationContext
khusus untuk operasi tak segerak dan pastikan semua operasi await
menggunakan konteks tersebut, mengelakkan konflik dengan urutan UI.
Atas ialah kandungan terperinci Mengapa Kaedah C# Asynchronous Saya Hang Apabila Mengakses Keputusan Tugasan?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!