2010-12-13 72 views
1

當客戶端連接到TCPListener時,我不斷收到以下異常。TCPListener異常的幫助

例外:

System.ObjectDisposedException:無法訪問已釋放的對象。

對象名稱:'System.Net.Sockets.NetworkStream'。

在System.Net.Sockets.NetworkStream.Read(字節[]緩衝液,的Int32偏移的Int32大小)

在Test.Server.ProcessClient(客戶對象,對象的clientId)


Server.cs

public class Server 
{ 
    private TcpListener tcpListener; 
    private Thread listenThread; 

    public event EventHandler<EventLogArgs> EventLog; 

    public Server() 
    { 
     // Startup Code 
     ThreadPool.SetMinThreads(50, 50); 
    } 

    void UpdateEventLog(EventLogArgs e) 
    { 
     if (EventLog != null) 
     { 
      EventLog(this, e); 
     } 
    } 

    public void Start(string ip, int port_num) 
    { 
     Globals.listen = true; 

     Int32 port = port_num; 
     IPAddress address = IPAddress.Parse(ip); 

     this.tcpListener = new TcpListener(address, port); 

     Socket listenerSocket = this.tcpListener.Server; 

     LingerOption lingerOption = new LingerOption(true, 10); 
     listenerSocket.SetSocketOption(SocketOptionLevel.Socket, 
          SocketOptionName.Linger, 
          lingerOption); 

     this.listenThread = new Thread(ListenForClients); 
     this.listenThread.Start(); 

     UpdateEventLog(new EventLogArgs("Started server...")); 
    } 

    public void Stop() 
    { 
     Globals.listen = false; 
     UpdateEventLog(new EventLogArgs("Stop server request sent...")); 
    } 

    private void ListenForClients() 
    { 
     this.tcpListener.Start(); 

     while (Globals.listen) 
     { 
      if (!this.tcpListener.Pending()) 
      { 
       // This is so we can stop the server. 
       Thread.Sleep(25); // choose a number (in milliseconds) that makes sense 
       continue; // skip to next iteration of loop 
      } 

      //blocks until a client has connected to the server 
      TcpClient client = this.tcpListener.AcceptTcpClient(); 

      Globals.clientRequests++; 
      int clientRequest = Globals.clientRequests; 
      UpdateEventLog(new EventLogArgs("(" + Globals.clientRequests + ") Client connected...\r\n")); 

      ThreadPool.QueueUserWorkItem(o => ProcessClient(client, clientRequest)); 
     } 

     UpdateEventLog(new EventLogArgs("Stopped server!")); 
     this.tcpListener.Stop(); 
    } 

    private void ProcessClient(object client, object clientId) 
    { 
     TcpClient tcpClient = (TcpClient)client; 
     int clientRequestId = (int)clientId; 
     NetworkStream clientStream = tcpClient.GetStream(); 

     byte[] clientRequestRaw = new byte[1024]; 
     int bytesRead; 

     UpdateEventLog(new EventLogArgs("(" + clientRequestId + ") Process client request...")); 

     while (true) 
     { 
      bytesRead = 0; 
      try 
      { 
       //blocks until a client sends a message 
       bytesRead = clientStream.Read(clientRequestRaw, 0, 512); 
      } 
      catch 
      { 
       //a socket error has occured 
       UpdateEventLog(new EventLogArgs("(" + clientRequestId + ") SOCKET ERROR\r\n\r\n" + e)); 
       break; 
      } 
      if (bytesRead == 0) 
      { 
       //the client has disconnected from the server 
       UpdateEventLog(new EventLogArgs("(" + clientRequestId + ") Client disconnected from server, nothing sent.")); 
       break; 
      } 

      //message has successfully been received. 
      ASCIIEncoding encoder = new ASCIIEncoding(); 
      string clientRequest = encoder.GetString(clientRequestRaw, 0, bytesRead); 

      string[] cmd; 
      string success; 
      Dictionary<string, string> headers = new Dictionary<string, string>(); 
      Dictionary<string, string> contents = new Dictionary<string, string>(); 

      if (clientRequest.Length == 0) 
      { 
       return; 
      } 

      // Parse HTTP request 
      Parse Parse = new Parse(); 
      Parse.HTTPRequest(clientRequest, out success, out cmd, out headers, out contents); 

      string response; 
      if (success == "TRUE") 
      { 
       response = "HTTP/1.1 200 OK\r\n\r\nHello World!\r\n"; 
      } 
      else 
      { 
       response = "HTTP/1.1 200 OK\r\n\r\nHello Error!\r\n"; 
      } 


      ResponseToClient(client, response); 
      clientStream.Close(); 

      UpdateEventLog(new EventLogArgs("(" + clientRequestId + ") Server response...\r\n\r\n" + response)); 
     } 

     tcpClient.Close(); 

     UpdateEventLog(new EventLogArgs("(" + clientRequestId + ") Client disconnected.")); 
    } 

    private void ResponseToClient(object client, string response) 
    { 
     TcpClient tcpClient = (TcpClient)client; 
     NetworkStream clientStream = tcpClient.GetStream(); 
     ASCIIEncoding encoder = new ASCIIEncoding(); 
     byte[] buffer = encoder.GetBytes(response); 

     clientStream.Write(buffer, 0, buffer.Length); 
     clientStream.Flush();    
    } 
} 

服務器類是從主線程開始像這樣:

this.server = new Server(); 
this.server.EventLog += new EventHandler<EventLogArgs>(UpdateEventLog); 
ThreadPool.QueueUserWorkItem(o => this.server.Start("127.0.0.1", 3000)); 

我在做什麼錯在這裏?

回答

0

您必須爲網絡提供正確的服務。
在services中寫services.msc,然後找到你的服務。 YourService->屬性 - >登錄 - >登錄爲網絡

1

我已經從ProcessClient()方法採取while循環,這似乎已經解決了這個問題。不管怎麼說,還是要謝謝你。

private void ProcessClient(object client, object clientId) 
{ 
    TcpClient tcpClient = (TcpClient)client; 
    int clientRequestId = (int)clientId; 
    NetworkStream clientStream = tcpClient.GetStream(); 

    byte[] clientRequestRaw = new byte[1024]; 
    int bytesRead; 

    UpdateEventLog(new EventLogArgs("(" + clientRequestId + ") Process client request...")); 

    while (true) 
    { 
     bytesRead = 0; 

     try 
     { 
      //blocks until a client sends a message 
      bytesRead = clientStream.Read(clientRequestRaw, 0, 512); 
     } 
     catch (Exception e) 
     { 
      //a socket error has occured 
      UpdateEventLog(new EventLogArgs("(" + clientRequestId + ") SOCKET ERROR\r\n\r\n" + e)); 
      break; 
     } 

     if (bytesRead == 0) 
     { 
      //the client has disconnected from the server 
      UpdateEventLog(new EventLogArgs("(" + clientRequestId + ") Client disconnected from server, nothing sent.")); 
      break; 
     } 

     //message has successfully been received. 
     ASCIIEncoding encoder = new ASCIIEncoding(); 
     string clientRequest = encoder.GetString(clientRequestRaw, 0, bytesRead); 

     string[] cmd; 
     string success; 
     Dictionary<string, string> headers = new Dictionary<string, string>(); 
     Dictionary<string, string> contents = new Dictionary<string, string>(); 

     if (clientRequest.Length == 0) 
     { 
      return; 
     } 

     // Parse HTTP request 
     Parse Parse = new Parse(); 
     Parse.HTTPRequest(clientRequest, out success, out cmd, out headers, out contents); 

     string response; 
     if (success == "TRUE") 
     { 
      response = "HTTP/1.1 200 OK\r\n\r\nHello World!\r\n"; 
     } 
     else 
     { 
      response = "HTTP/1.1 200 OK\r\n\r\nHello Error!\r\n"; 
     } 

     ResponseToClient(client, response); 

     clientStream.Close(); 

     UpdateEventLog(new EventLogArgs("(" + clientRequestId + ") Server response...\r\n\r\n" + response)); 
    } 

    tcpClient.Close(); 

    UpdateEventLog(new EventLogArgs("(" + clientRequestId + ") Client disconnected.")); 
} 
0

既然你解決你的問題,我只是要給你兩個提示:

首先,不使用線程池進行同步操作異步的。這是浪費資源。改用異步方法(BeginAccept/EndAccept等)。

接下來將您的課程分爲多個課程。一個只關心服務器部件,一個只關心客戶端部件。它使你的代碼更容易遵循(即使對你:))。

相關問題