Memahami setiap Gelung dan Penutupan dalam C# Berbilang Thread
Multithreading dalam C# boleh memperkenalkan kerumitan apabila menggunakan foreach
gelung dan penutupan. Masalah timbul apabila penutupan dalam gelung foreach
merujuk pembolehubah lelaran gelung.
Pertimbangkan contoh ini:
<code class="language-csharp">var threads = new List<Thread>(); foreach (Foo f in ListOfFoo) { Thread thread = new Thread(() => f.DoSomething()); threads.Add(thread); thread.Start(); }</code>
Di sini, setiap penutupan urutan menangkap rujukan kepada pembolehubah f
tunggal. Semasa gelung berlangsung, nilai f
berubah, membawa kepada hasil yang tidak dapat diramalkan—benang mungkin beroperasi pada objek Foo
yang tidak dijangka.
Memastikan Keselamatan Benang
Untuk mengelakkan perkara ini, buat pembolehubah tempatan baharu dalam setiap lelaran:
<code class="language-csharp">var threads = new List<Thread>(); foreach (Foo f in ListOfFoo) { Foo f2 = f; // Create a copy for each thread Thread thread = new Thread(() => f2.DoSomething()); threads.Add(thread); thread.Start(); }</code>
Kini, setiap urutan mempunyai rujukan bebas sendiri (f2
), memastikan DoSomething()
dipanggil pada contoh Foo
yang betul.
C# 5 dan Seterusnya
Tingkah laku berubah sedikit dalam C# 5 dan kemudian. Pengkompil menganggap pembolehubah foreach
sebagai diisytiharkan secara berkesan dalam skop gelung, walaupun ia muncul di luar. Oleh itu, kedua-dua coretan kod di atas adalah selamat untuk benang dalam C# 5 dan versi seterusnya. Walau bagaimanapun, salinan eksplisit itu kekal sebagai amalan yang baik untuk kejelasan dan keserasian ke belakang.
Atas ialah kandungan terperinci Mengapa Gelung dan Penutupan Foreach Memerlukan Pengendalian Khas dalam Kod C# Berbilang Benang?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!