2010-02-20 216 views
2

我正在寫一個異步服務器和客戶端與C#。我從MSDN獲取了示例代碼,並進行了修改,並使其多次發送和接收消息。我想5個多的客戶連接到服務器,它的工作,但在第98次迭代每個客戶拋出異常的StackOverflow。有人可以解釋我爲什麼會出現這個錯誤嗎?我已經閱讀MSDN,我的問題在無限循環,但我不明白我該如何更改我的代碼,並寫入它沒有循環。C#,異步套接字服務器/客戶端,StackOverflowException

此服務器的客戶端應在多人賽車遊戲中使用,我需要每秒每個球員的發送和接收的座標幾次。我怎樣才能做到沒有無限循環?

這裏是我的代碼:

SERVER:

using System; 
using System.Collections.Generic; 
using System.Linq; 
using System.Text; 
using System.Net; 
using System.Net.Sockets; 
using System.Threading; 


public class Server 
{ 
    Socket main_tcp_Sock; 
    private static ManualResetEvent acceptDone = new ManualResetEvent(false); 
    private static ManualResetEvent sendDone = new ManualResetEvent(false); 
    private static ManualResetEvent recvDone = new ManualResetEvent(false); 
    private static ManualResetEvent closeDone = new ManualResetEvent(false); 
    //int cl_Count = 0; 
    List<StateObject> connection_List = new List<StateObject>(); 
    private static String response = String.Empty; 


    public class StateObject 
    { 
     public Socket current_Socket = null; 
     public byte[] data = new byte[256]; 
     public string id = string.Empty; 
    } 
    public Server() 
    { 
     Server_Start(); 
    } 

    public void Server_Start() 
    { 

     //Creating socket 
     main_tcp_Sock = new Socket(AddressFamily.InterNetwork, 
            SocketType.Stream, 
            ProtocolType.Tcp); 
     IPEndPoint ipLocal = new IPEndPoint(IPAddress.Parse("127.0.0.1"), 2000); 
     //Bind socket 
     try 
     { 
      main_tcp_Sock.Bind(ipLocal); 
      Console.WriteLine("Server has started successfully!"); 


      //Start listening 
      main_tcp_Sock.Listen(100); 
      while (true) 
      { 

       acceptDone.Reset(); 
       Console.WriteLine("Waiting for a connection..."); 

       //AsyncAccept 
       main_tcp_Sock.BeginAccept(new AsyncCallback(On_Connect), main_tcp_Sock); 
       acceptDone.WaitOne(); 
       Console.WriteLine("\nPress any button to continue...\n\n"); 
       Console.ReadKey(true); 
      } 
     } 

     catch (Exception e) 
     { 
      Console.WriteLine(e.ToString()); 
      Console.ReadKey(true); 
     } 

    } 

    public void On_Connect(IAsyncResult asyn) 
    { 

     try 
     { 

      Socket listener = (Socket)asyn.AsyncState; 
      Socket handler = listener.EndAccept(asyn); 
      acceptDone.Set(); 

      StateObject connection = new StateObject(); 
      connection.current_Socket = handler; 

      if (!connection_List.Contains(connection)) 
      { 
       lock (connection_List) 
       { 
        connection_List.Add(connection); 
        connection.id = "00" + connection_List.Count.ToString() + " "; 
       } 
      } 
      recvDone.Reset(); 
      Receive(connection.current_Socket); 
      recvDone.WaitOne(); 

      sendDone.Reset(); 
      Send(connection.current_Socket, response); 
      sendDone.WaitOne(); 

      closeDone.Reset(); 
      Socket_Close(connection.current_Socket); 
      closeDone.WaitOne(); 
     } 

     catch (Exception e) 
     { 
      Console.WriteLine("On_Connect Error: {0}", e.ToString()); 
      Console.ReadKey(true); 
     } 
    } 

    public void Receive(Socket handler) 
     { 
     try{ 
      StateObject connection = new StateObject(); 
      connection.current_Socket = handler; 
      connection.current_Socket.BeginReceive(connection.data, 0, connection.data.Length, 0, 
       new AsyncCallback(On_Receive), connection); 
     } 
     catch (Exception e){ 
     Console.WriteLine(e.ToString()); 
      Console.ReadKey(true); 
     } 

    } 
    public void On_Receive(IAsyncResult asyn) 
    { 
     string content = ""; 
     string temp = ""; 
     StateObject connection = (StateObject)asyn.AsyncState; 
     Socket handler = connection.current_Socket; 
     int size = handler.EndReceive(asyn); 


     Console.WriteLine("ConnID from receive: " + connection.id); 
     if (size > 0) 
     { 
      temp += Encoding.ASCII.GetString(connection.data); 
     } 
     if (temp.IndexOf("<EOF>") > -1) 
     { 

      content += temp.Substring(0, temp.IndexOf("\0")); 
      Console.WriteLine("Read {0} bytes from socket. \nMessage: {1}", content.Length, content); 

      lock (connection_List) 
      { 
       foreach (StateObject conn in connection_List) 
       { 
        if (conn != connection) 
        { 
        content.Insert(0, connection.id); 
        response = content; 
        } 

       } 
      } 
      recvDone.Set(); 
     } 
     else 
     { 
      handler.BeginReceive(connection.data, 0, connection.data.Length, 0, new AsyncCallback(On_Receive), connection); 
     } 

    } 

    public void Send(Socket handler, String message) 
    { 
     byte[] data = Encoding.ASCII.GetBytes(message); 
     handler.BeginSend(data, 0, data.Length, 0, new AsyncCallback(On_Send), handler); 

    } 

    public void On_Send(IAsyncResult result) 
    { 
     try 
     { 
      StateObject state = new StateObject(); 
      Socket handler = (Socket)result.AsyncState; 
      state.current_Socket = handler; 
      int size = state.current_Socket.EndSend(result); 
      if (size > 0) 
      { 
       sendDone.Set(); 
      } 

      else state.current_Socket.BeginSend(state.data, 0, state.data.Length, SocketFlags.None, 
       new AsyncCallback(On_Send), state); 
      Console.WriteLine("Bytes sent to client: {0}", size); 

      sendDone.Set(); 
     } 

     catch (Exception e) 
     { 
      Console.WriteLine("On_Send e, error: " + e.ToString()); 
      Console.ReadKey(true); 
     } 
    } 
    public void Socket_Close(Socket sock) 
    { 
     sock.LingerState = new LingerOption(true, 3); 

     sock.Shutdown(SocketShutdown.Both); 
     sock.Close(); 
     closeDone.Set(); 
    } 

} 

和客戶端:

using System; 
using System.Net; 
using System.Net.Sockets; 
using System.Threading; 
using System.Text; 

// State object for receiving data from remote device. 
public class StateObject 
{ 
    // Client socket. 
    public Socket workSocket = null; 
    // Size of receive buffer. 
    public const int BufferSize = 256; 
    // Receive buffer. 
    public byte[] buffer = new byte[BufferSize]; 
    // Received data string. 
    public StringBuilder sb = new StringBuilder(); 
} 

public class AsynchronousClient 
{ 
    public static int count = 0; 

    // The port number for the remote device. 
    private const int port = 2000; 
    // ManualResetEvent instances signal completion. 
    private static ManualResetEvent connectDone = new ManualResetEvent(false); 
    private static ManualResetEvent sendDone = new ManualResetEvent(false); 
    private static ManualResetEvent receiveDone = new ManualResetEvent(false); 
    private static ManualResetEvent closeDone = new ManualResetEvent(false); 
    // The response from the remote device. 
    private static String response = String.Empty; 

    private static void StartClient() 
    { 
     // Connect to a remote device. 

     IPEndPoint remoteEP = new IPEndPoint(IPAddress.Parse("127.0.0.1"), port); 
     // Create a TCP/IP socket. 
     Socket client = new Socket(AddressFamily.InterNetwork, 
      SocketType.Stream, ProtocolType.Tcp); 
     Start(client, remoteEP); 

    } 


    public static void Start(Socket client, EndPoint remoteEP) 
    { 
     try 
     { 
      while (true) 
      { 
       /*if (count >= 30) 
       { 

        Thread.Sleep(1000); 
        if (count >= 100) 
        { 
         count = 0; 
         Thread.Sleep(1500); 
        } 
       }*/ 

       Console.WriteLine(count); 
       connectDone.Reset(); 
       client.BeginConnect(remoteEP, new AsyncCallback(ConnectCallback), client); 
       connectDone.WaitOne(); 


       // Send test data to the remote device. 
       sendDone.Reset(); 
       Send(client, "Some text and <EOF>"); 
       sendDone.WaitOne(); 

       // Receive the response from the remote device. 
       receiveDone.Reset(); 
       Receive(client); 
       receiveDone.WaitOne(); 

       // Write the response to the console. 
       Console.WriteLine("Response received : {0}", response); 

       // Release the socket. 
       closeDone.Reset(); 
       Socket_Close(client); 
       closeDone.WaitOne(); 

       ++count; 
      } 

     } 
     catch (ObjectDisposedException) 
     { 
      Socket sock = new Socket(AddressFamily.InterNetwork, 
        SocketType.Stream, ProtocolType.Tcp); 
      IPEndPoint remote = new IPEndPoint(IPAddress.Parse("127.0.0.1"), port); 

      Start(sock, remote); 

     } 
     catch (SocketException) 
     { 
      Socket sock = new Socket(AddressFamily.InterNetwork, 
         SocketType.Stream, ProtocolType.Tcp); 
      IPEndPoint remote = new IPEndPoint(IPAddress.Parse("127.0.0.1"), port); 

      Start(sock, remote); 

     } 
     catch (Exception e) 
     { 
      Console.WriteLine(e.ToString()); 
      Console.ReadKey(true); 
     } 
    } 

    private static void ConnectCallback(IAsyncResult ar) 
    { 
     try 
     { 
      // Retrieve the socket from the state object. 
      Socket client = (Socket)ar.AsyncState; 

      // Complete the connection. 
      client.EndConnect(ar); 

      Console.WriteLine("Socket connected to {0}", 
       client.RemoteEndPoint.ToString()); 

      // Signal that the connection has been made. 
      connectDone.Set(); 
     } 
     catch (Exception e) 
     { 
      Console.WriteLine(e.ToString()); 
      Console.ReadKey(true); 
     } 
    } 

    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()); 
      Console.ReadKey(true); 

     } 
    } 

    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()); 
      Console.ReadKey(true); 

     } 
    } 

    private static void Send(Socket client, String data) 
    { 
     try 
     { 
      // Convert the string data to byte data using ASCII encoding. 
      byte[] byteData = Encoding.ASCII.GetBytes(data); 

      // Begin sending the data to the remote device. 
      client.BeginSend(byteData, 0, byteData.Length, 0, 
       new AsyncCallback(SendCallback), client); 
     } 
     catch (Exception e) 
     { 
      Console.WriteLine(e.ToString()); 
      Console.ReadKey(true); 
     } 
    } 

    private static void SendCallback(IAsyncResult ar) 
    { 
     try 
     { 
      // Retrieve the socket from the state object. 
      Socket client = (Socket)ar.AsyncState; 

      // Complete sending the data to the remote device. 
      int bytesSent = client.EndSend(ar); 
      Console.WriteLine("Sent {0} bytes to server.", bytesSent); 

      // Signal that all bytes have been sent. 
      sendDone.Set(); 
     } 
     catch (Exception e) 
     { 
      Console.WriteLine(e.ToString()); 
      Console.ReadKey(true); 
     } 
    } 

    public static void Socket_Close(Socket sock) 
    { 
     try 
     { 
      sock.LingerState = new LingerOption(true, 3); 
      sock.Shutdown(SocketShutdown.Both); 
      sock.Close(); 
      closeDone.Set(); 
     } 
     catch (Exception e) 
     { 
      Console.WriteLine(e.ToString()); 
      Console.ReadKey(true); 
     } 
    } 

    public static int Main(String[] args) 
    { 

     StartClient(); 
     return 0; 
    } 
} 

我在C#中新。請幫助別人。

+1

而不是在while循環中監聽,您也可以使用計時器。 – Adeel 2010-02-20 16:16:31

+0

感謝您的回覆,我會在C#中閱讀關於計時器的內容,但是您能否解釋我有什麼不同?爲什麼循環帶來了Stackoverflow,但定時器不? – 2010-02-20 17:17:45

回答

1

while (true)將永遠不會停止......你必須這樣做while (foo) { if (somecondition) foo=false }

編輯:也許本教程將幫助

http://www.codeguru.com/csharp/csharp/cs_date_time/timeroutines/article.php/c7763

片段:

using System; 
using System.Timers; 

class myApp 
{ 
    public static void Main() 
    { 
    Timer myTimer = new Timer(); 
    myTimer.Elapsed += new ElapsedEventHandler(DisplayTimeEvent); 
    myTimer.Interval = 1000; 
    myTimer.Start(); 

    while (Console.Read() != 'q') 
    { 
     ; // do nothing... 
    } 
    } 

    public static void DisplayTimeEvent(object source, ElapsedEventArgs e) 
    { 
     Console.Write("\r{0}", DateTime.Now); 
    } 
} 
+0

是的,我知道。當玩家退出遊戲時,這個循環將停止,但它在玩遊戲時應該起作用,因爲它應該發送他自己的座標並且每秒接收幾次其他玩家的座標。這是不可能的循環?計時器是否更好?爲什麼?我只是想了解它。謝謝。 – 2010-02-20 17:32:53

+0

是的......一個'while'循環會阻塞線程(直到它退出時什麼都不會發生)......你應該設置一個計時器來檢查每N毫秒......我可以提供一些ActionScript等效代碼(不是C#傢伙) – mga 2010-02-20 17:48:41

+0

我想用計時器重寫我的代碼,但它不起作用。我會嘗試更多。謝謝。 – 2010-02-20 20:44:43

1

堆棧是一個後進先出隊列。在進行函數調用時,需要保存與所在函數(例如局部變量)相關的狀態,以便程序可以處理被調用的函數。當被調用函數返回時,調用函數的狀態必須被恢復,所以它從堆棧中被檢索。在普通程序中,隨着您調用越來越深的嵌套函數,堆棧會不斷增長,但隨着它們的返回而再次縮小。

當然,堆棧並不是無限的,實際上通常會分配一個固定的內存量,所以如果你有一個嵌套函數調用級別不斷增長的程序,堆棧最終會滿載,而你會收到堆棧溢出異常。

證明這一點最簡單的方法是編寫一個函數總是遞歸調用自己。下面將始終創建一個堆棧溢出:

private void blowTheStack(){ 
    blowTheStack(); 
} 

當然,我們也可以編寫遞歸程序沒有這個問題:

private void doSomethingUseful(){ 
    if (!terminatingConditionReached()){ 
    doSomethingUseful(); 
    } 
} 

,這是一個非常強大的技術。編寫這樣的代碼的技巧是讓你的終止條件正確!

當這一切發生在一個功能,邏輯是很容易看到,但是,你也可以有類似的邏輯,其中一個函數調用另一個,它調用另一個,它調用第一個函數。

當你和事件處理代碼的工作,還有更糟糕的,因爲函數調用並不像函數調用了。假設一個函數觸發一個觸發事件處理程序的條件,並且事件處理程序觸發一些調用第一個函數的代碼,您也可以使用循環函數調用,並且如果終止條件不正確,則會發生堆棧溢出。

無論如何 - 我只是想+1這個問題,因爲它是關於堆棧溢出!