两个不同的套接字实例可以侦听相同的TCP端口(端口已在使用中)

时间:2015-01-26 14:24:59

标签: java sockets tcp

我有一个TcpServer类负责,就像tcp服务器一样。你可以在下面找到这个课程:

public class TcpServer {
    private ServerSocket serverSocket;
    private Socket socket;
    private int locallyBoundPort;

    public TcpServer() {


    }

    public TcpServer(int locallyBoundPort) {
        try {
            this.serverSocket = new ServerSocket(locallyBoundPort);
            serverSocket.setReuseAddress(true);
        } catch (IOException e) {
            System.out.println("Error at binding to port TCP : " + locallyBoundPort + "...cause : " + e.getMessage());
        }
        socket = null;

    }

    public void accept() {
        try {
            socket = serverSocket.accept();
            socket.setReuseAddress(true);
        } catch (IOException e) {
            System.out.println("Error at accept : " + locallyBoundPort);
        }
    }



    public void send(Data data) throws IOException {
        if(socket != null) {

            ObjectOutputStream out = new ObjectOutputStream(socket.getOutputStream());
            out.writeObject(data);

        }
    }

    public Data receive() throws ClassNotFoundException, IOException {
        if(socket != null) {

                ObjectInputStream in = new ObjectInputStream(socket.getInputStream());
                return (Data) in.readObject();

        } else {
            return null;
        }
    }

    public boolean bind(int port) throws IOException {
        try {
            this.serverSocket = new ServerSocket(port);
            this.locallyBoundPort = port;
        } catch(IOException e) {
            return false;
        }
        return true;

    }

    public void close() {
        try {
            serverSocket.close();
            socket.close();
        } catch (IOException e) {
            OzumUtils.print("IOException in close, TcpServer");
        }

    }

    public int getLocallyBoundPort() {
        return locallyBoundPort;
    }

    public Socket getSocket() {
        return socket;
    }

    public ServerSocket getServerSocket() {
        return serverSocket;
    }
}

我有一个代码片可以做到这一点:

TcpServer tcpServer = new TcpServer(LocalPort);
while(1)
{
    tcpServer.accept();
    Thread thread = new Thread(new runnable(tcpServer));
    thread.start();
    tcpServer = new TcpServer(LocalPort);
}

但是我得到一个端口已经在使用错误。我认为两个不同的套接字实例可以监听同一个端口,因为当连接器有不同的ip或端口时,多路复用允许两个连接通过同一个端口? 我错过了什么?

2 个答案:

答案 0 :(得分:5)

您无法将两个tcp服务器套接字绑定到同一端口。 reuseAddress实际上是用于客户端套接字,并且它不像您认为的那样工作......并且您使用它的方式根本不会做任何事情(因为您在绑定后设置它)。

你真的不需要两次绑定到同一个端口。只需从tcpServer = new TcpServer(LocalPort);循环底部删除此行while,即可完成所有设置。

这种方式的工作方式是绑定服务器套接字一次并监听端口。当连接到达时,它会分叉客户端套接字以便您与客户端通信,并且原始服务器套接字继续侦听更多连接。

基本上,您需要从socket中删除TcpServer成员(以及任何其他状态),并使accept方法返回已接受的套接字。然后让runnablesocket作为参数而不是TcpServer,并使用它来提供客户端连接。然后继续在循环中调用accept,并以你知道的方式为新连接分支线程,除非每次都不重新创建服务器。

或者,从TcpServer中移除服务器套接字和端口,在循环外创建套接字,然后在其上while(true)调用accept,创建一个新的TcpServer使用返回的客户端套接字,并在线程中使用它来处理连接。

完成后,请不要忘记关闭客户端套接字。

答案 1 :(得分:1)

不,您不能使用已处于侦听状态的端口。但是,任意数量的客户端都可以连接到同一个端口。您不需要再次侦听端口,只需生成一个新线程来处理当前连接并等待新连接。例如,假设您有一个实现TcpConnectionHanlder的类Runnable并将Socket作为参数,则循环看起来像

while (true) { //while(1) is not valid Java syntax
    final Socket tcpSocket = tcpServer.accept(); // Get socket for incoming connection
    final Thread thread = new Thread(new TcpConnectionHanlder(tcpSocket)); // Create a thread for this socket/client connection
    thread.start(); // Launch the thread
    // tcpServer = new TcpServer(LocalPort); <- not needed, port still listening.
}

然后在你的TcpConnectionHanlder实例中处理这个特定的客户端(套接字)。