为什么在.Net中使用异步编程模型不会导致StackOverflow异常?

uri*_*ium 5 .net asynchronous

例如,我们调用BeginReceive并具有BeginReceive完成后执行的回调方法.如果那个回调方法再次在我的脑海中调用BeginReceive,那将非常类似于递归.这怎么会导致stackoverflow异常.来自MSDN的示例代码:

private static void Receive(Socket client) {
    try {
        // Create the state object.
        StateObject state = new StateObject();
        state.workSocket = client;

        // Begin receiving the data from the remote device.
        client.BeginReceive( state.buffer, 0, StateObject.BufferSize, 0,
            new AsyncCallback(ReceiveCallback), state);
    } catch (Exception e) {
        Console.WriteLine(e.ToString());
    }
}

private static void ReceiveCallback( IAsyncResult ar ) {
    try {
        // Retrieve the state object and the client socket 
        // from the asynchronous state object.
        StateObject state = (StateObject) ar.AsyncState;
        Socket client = state.workSocket;

        // Read data from the remote device.
        int bytesRead = client.EndReceive(ar);

        if (bytesRead > 0) {
            // There might be more data, so store the data received so far.
        state.sb.Append(Encoding.ASCII.GetString(state.buffer,0,bytesRead));

            // Get the rest of the data.
            client.BeginReceive(state.buffer,0,StateObject.BufferSize,0,
                new AsyncCallback(ReceiveCallback), state);
        } else {
            // All the data has arrived; put it in response.
            if (state.sb.Length > 1) {
                response = state.sb.ToString();
            }
            // Signal that all bytes have been received.
            receiveDone.Set();
        }
    } catch (Exception e) {
        Console.WriteLine(e.ToString());
    }
}
Run Code Online (Sandbox Code Playgroud)

MK.*_*MK. 3

BeginReceive 注册与重叠 IO 操作关联的回调函数。当数据可用时,操作系统将调用回调,但 BeginReceive 调用会立即返回,因此您对 ReceiveCallback 的调用也会完成。
将实际 IO 视为发生在不属于您、而是属于操作系统的线程中。您注册回调的行为只是说“发生事情时继续打电话给我”,但它不会添加到堆栈中。这就是为什么它被称为异步。