以SOCKET通信中的异步方法为例:
public static ManualResetEvent ConnectDone = new ManualResetEvent(false);
公共静态无效 ConnectCallback(IAsyncResult ar)
{
Socket sClient = (Socket)ar.AsyncState;
sClient.EndConnect(ar);
Console.WriteLine("套接字已连接到 {0}", sClient.RemoteEndPoint.ToString());
ConnectDone.Set();
}
公共静态无效Main(字符串[] arg)
{
尝试
{
IPHostEntry ipHost = Dns.Resolve(“127.0.0.1”);
IPAddress ipAddr = ipHost.AddressList[0];
IPEndPoint endPoint = new IPEndPoint(ipAddr, 11000);
Socket sClient = new Socket(AddressFamily.InterNetwork,
SocketType.Stream, ProtocolType.Tcp);
sClient.BeginConnect(endPoint, new AsyncCallback(ConnectCallback),
sClient);
for (int i = 0; i <5; i++)
Console.WriteLine("做一些其他工作。");
byte[] byteData = Encoding.ASCII.GetBytes("一些数据。");
ConnectDone.WaitOne();
sClient.BeginSend(byteData, 0, byteData.Length, 0,
new AsyncCallback(SendCallback), sClient);
……
}
(注:ManualResetEvent 允许线程通过发送信号相互通信。通常,此通信涉及一个线程在其他线程进行之前必须完成的任务。ManualResetEvent
就像一个信号灯,可以利用它的信号来通知其他线程。它有几个重要的地方的方法:Reset(),Set(),WaitOne()。初始化该对象时,用户可以指定其默认的状态(有信号/无信号),在初始化以后,该对象将保持原来的状态不变直到它的Reset()或者Set()方法被调用,Reset()方法将其设置为无信号状态,Set()方法将其设置为有信号状态。WaitOne()方法使当前线程挂起直到ManualResetEvent对象有一个信号状态,此时该线程将被激活。)
本例中,主线程调用ConnectDone.WaitOne();之后,主线程阻塞,直到连接操作(即ConnectCallback)完成,因为连接操作完成之后,执行了ConnectDone.Set()将信号灯设置为有信号,由于调用了ManualResetEvent.WaitOne()方法而初始化等待状态的主线程将接收到这个信号,于是它继续往下执行,完成后面的工作。
同步与异步:
简单来说,同步即程序执行一个方法,等该方法返回之后,继续往下走,
异步:即程序调用一个方法后立即返回,“宏”而言,主线程与方法线程线程执行
就本例而言,socket的异步方法BeginConnect被调用后,继续执行主线程中该语句之后的代码,即:
for (int
i = 0; i <5; i++)
Console.WriteLine("Do Some Other工作。”);
byte[] byteData = Encoding.ASCII.GetBytes("Some Data.");
如果调用的是同步方法,那么,必须输出“Socket连接到…”一定会在“Do Some Other Work.”之前,因为主线程等待同步方法的返回,但是在异步的情况下,将可能在 5 行“Do Some Other Work.”之间的某个时刻出现“Socket 连接到...”的输出(实际情况还是依赖于网络的线程调度)。
至于在 sClient.BeginSend(…) 方法之前调用 ConnectDone.WaitOne(),否则此前依赖于连接操作(即 ConnectCallback)的完成,须同步一下。