我正在创建一个聊天服务器,但无法建立多个连接,我相信这是因为只有一个套接字。 我读过,一个线程一次可以处理多个套接字连接,但是每个客户端必须拥有自己的套接字。 像墙上的插座。 每个插头必须有1个插座。 如何为每个客户端创建用于与服务器通信的套接字? 我将提供一些代码来帮助你们以帮助为基础。 这是接受单个套接字(连接)的地方。

 private void waitForConnection() throws IOException
 {
     connection = server.accept();
     showMessage("\nNow connected to " + connection.getInetAddress().toString());
     clientCount += 1;
 }

这是从服务器启动处调用代码的位置:

public void startRunning()
{
  try
  {
     server = new ServerSocket(1337, 10);
     while(true)
     {
        try
        {
            showMessage("Waiting For a Player To Connect...");
            waitForConnection();
            setupStreams();
            whileChatting();

        }
        catch(EOFException eofException)
        {
           showMessage("\n Server ended the connection! ");
        }
        finally
        {
           closeConnections();
        }
     }
  }
  catch(IOException ioException)
  {
     ioException.printStackTrace();
  }
}

现在,此代码对于第一个已连接的客户端非常有效。 所有其他人根本无法连接。 在此先感谢所有提交答案的人。

===============>>#1 票数:1

您应该让一个线程始终执行socket.accept() 那是:

  • 每当您收到客户端(通过socket.accept() )时,都应该
    • 启动一个新的Thread ,从现在开始将负责该客户端。 (我将这个Thread称为扩展类ClientNanny );
    • 然后再次运行socket.accept()

这是您的waitForConnection()方法的外观:

private List<ClientNanny> clients = new ArrayList<ClientNanny>();

private void waitForConnection() throws IOException
{
     while (true) { /* or some other condition you wish */
         connection = server.accept(); /* will wait here */
         /* this code is executed when a client connects... */

         showMessage("\nNow connected to " 
                                      + connection.getInetAddress().toString());

         ClientNanny nanny = new ClientNanny(connection); /* call a nanny to take
                                                             care of it */
         clients.add(nanny); /* make sure you keep a ref to it, just in case */
         nanny.start(); /* tell nanny to get to work as an independent thread */
         clientCount += 1; /* you dont need this: use clients.size() now */
     }
}

如果您没有注意到, ClientNanny应该类似于:

class ClientNanny extends Thread {
    ClientNanny(Socket baby) {
        this.myBaby = baby;
    }
    @Override
    public void run() {
        bed.put(myBaby); /* and probably other commands :) */
    }
}

现在,您应该注意到下面突出显示的代码...

try
{
    showMessage("Waiting For a Player To Connect...");
    waitForConnection();
    setupStreams();  /* <--------- THIS LINE! HI! I'M A HIGHLIGHT! */
    whileChatting(); /* <--------- THIS LINE! HI! I'M A HIGHLIGHT TOO! */
}

当您按照我的建议更改waitForConnection()时,将不会执行(因为waitForConnection()将在while(true)循环内不断滚动)。 也许您可以将socket.accept()放在其自己的线程上...无论如何,我相信您可以从现在开始使用它,是吗?

PS .:作为最终指针,上述解决方案使用new Thread() (或与此相关的new ClientNanny() ),因为这是解释问题的最简单方法。 但是,非常理想的解决方案可能涉及使用ExecutorService (大致是线程池),正如在此答案中迅速指出的那样。

===============>>#2 票数:0

您的代码的整个结构是错误的:

waitForConnection();
setupStreams();
whileChatting();

这不能满足您的要求。 您的循环应该不做任何事情,只接受连接,然后为处理该套接字上所有I / O的每个线程启动一个新线程。 请参阅Java教程的“ 定制网络”部分。

===============>>#3 票数:0

您的套接字代码应类似于以下PSEUDO-CODE:

create server socket
while true (or quit received) {
    accept connection
    spawn thread to handle communication
}

也就是说,您需要为每个连接生成一个线程。 然后, 每个线程都可以执行whileChatting() ,而不用占用主线程。

例:

while (true) {
    try {
        // listen for client connections
        Socket s = svr.accept();

        // start a separate thread to handle this client's communications
        new ChatClientHandler(s).start();
    }
    catch (IOException e) {
        // print out the error, but continue!
        System.err.println(e);
    }
}

  ask by user2278118 translate from so

未解决问题?本站智能推荐: