作者:陈浩颖美娇承湖_527 | 来源:互联网 | 2023-12-14 15:56
本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。
在Console中工作但在Windows窗体中不工作的异步代码
我正在尝试编写一个不断在局域网上搜索主机的应用程序。 当我将此作为控制台运行时,countdown.Wait()似乎工作正常。 但是,当我将代码带入windows窗体时,countdown.Signal()似乎没有减少它的计数器。 不确定是什么问题。
using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System.Text; using System.Windows.Forms; using System.Net.NetworkInformation; using System.Diagnostics; using System.Net; using System.Threading; namespace Multi_Threaded { public partial class Form1 : Form { public Form1() { InitializeComponent(); } private void Form1_Load(object sender, EventArgs e) { PortScanner ps = new PortScanner(); ps.ProbeCompleted += new PingProbeCompleted(ps_ProbeCompleted); ps.run_ping_probe(); } void ps_ProbeCompleted(object sender, PingProbeCompletedArguments e) { MessageBox.Show("I found " + e.ip_adresses_list_of_host.Count.ToString() + "host(s)"); } } public delegate void PingProbeCompleted(object sender,PingProbeCompletedArguments e); public class PingProbeCompletedArguments : EventArgs { public List ip_adresses_list_of_host; } public class PortScanner { public event PingProbeCompleted ProbeCompleted; static List ip_adresses = new List(); static CountdownEvent countdown; public void run_ping_probe() { ip_adresses.Clear(); countdown = new CountdownEvent(1); string ipBase = "10.125."; for (int sub = 0; sub <14; sub++) { for (int i = 1; i <255; i++) { string ip = ipBase + sub.ToString() + "." + i.ToString(); Ping p = new Ping(); p.PingCompleted += new PingCompletedEventHandler(p_PingCompleted); countdown.AddCount(); p.SendAsync(ip, 100, ip); } } countdown.Signal(); countdown.Wait(); PingProbeCompletedArguments e = new PingProbeCompletedArguments(); e.ip_adresses_list_of_host = ip_adresses; ProbeCompleted(this, e); } private void p_PingCompleted(object sender, PingCompletedEventArgs e) { string ip = (string)e.UserState; if (e.Reply.Status == IPStatus.Success) { ip_adresses.Add(ip + "t" + e.Reply.RoundtripTime + " ms"); } countdown.Signal(); } }
是的,当您在Winforms项目中使用它时,代码会死锁。 问题是Ping类尽力在调用SendAsync()的同一线程上引发PingCompleted事件。 它使用AsyncOperationManager.CreateOperation()方法来执行此操作。
问题是,这实际上适用于Winforms应用程序。 它试图在主线程上引发事件。 但是,由于您使用countdown.Wait()调用阻止了主线程,因此无法工作。 由于主线程被阻止,ping无法完成。 由于ping未完成,主线程无法完成。 死锁城市。
它在控制台模式应用程序中工作,因为它没有像Winforms那样的同步提供程序。 PingComplete事件将在线程池线程上引发。
阻止UI线程从根本上是有缺陷的。 快速解决方法是在工作线程上运行代码。 请注意,这也会导致ProbeCompleted事件在该worker上触发。 使用Control.BeginInvoke()将其封送到UI线程。 或者使用BackgroundWorker。
private void Form1_Load(object sender, EventArgs e) { PortScanner ps = new PortScanner(); ps.ProbeCompleted += new PingProbeCompleted(ps_ProbeCompleted); ThreadPool.QueueUserWorkItem((w) => ps.run_ping_probe()); }
并且不要忘记删除额外的Signal()调用。
你的等待处理程序在线程池的一个线程下运行。 你需要回到UI线程来更新UI(由于运行UI的消息循环) &#8211; 为此你使用SynchronizationContext
有关如何使用它的更多信息: http : //www.codeproject.com/KB/threads/SynchronizationContext.aspx
上述就是C#学习教程:在Console中工作但在Windows窗体中不工作的异步代码分享的全部内容,如果对大家有所用处且需要了解更多关于C#学习教程,希望大家多多关注&#8212;编程笔记