我正在创建一个基于Java的服务器。
我正在使用服务器套接字接受传入消息。
然而,在我的程序中的某些时候,我希望服务器套接字能够侦听另一个端口。
我关闭了服务器套接字。用我的新端口开始一个新的。一切都很好。
然而,当我再次将服务器套接字更改为上一个端口时,它会给我一个错误。
我已经读过服务器套接字在关闭它之后在超时状态下保持一段时间的事情。
所以这是我的问题: 我可以绕过服务器套接字的这个超时状态,并在关闭它之后再次使我的端口可用并想再次收听同一个端口吗?
public void makeServerSocketWithPort(int portnr) throws IOException, Exception
{
server = new ServerSocket(portnr);
server.setReuseAddress(true);
while(!portchanged)
{
Socket sock = server.accept();
System.out.println(server.getLocalPort());
System.out.println(sock.getLocalPort());
handler = new Requesthandler(sock); //should be in a thread
System.out.println(server.getLocalPort());
System.out.println(sock.getLocalPort());
}
}
public void invalidateRequestHandler(int newPort)
{
if(server != null)
{
portchanged = true;
try {
server.close();
} catch (IOException ex) {
Logger.getLogger(Controlserver.class.getName()).log(Level.SEVERE, null, ex);
}
}
portchanged = false;
makeServerSocketWithPort(newPort);
}
错误StackTrace:
Exception in thread "main" java.net.SocketException: Socket closed
at java.net.PlainSocketImpl.socketAccept(Native Method)
at java.net.PlainSocketImpl.accept(PlainSocketImpl.java:408)
at java.net.ServerSocket.implAccept(ServerSocket.java:462)
at java.net.ServerSocket.accept(ServerSocket.java:430)
at stuff.Controlserver.makeServerSocketWithPort(Controlserver.java:63)
at stuff.Main.main(Main.java:44)
编辑: 第二次尝试修复它无济于事:
public void makeServerSocketWithPort(int portnr, boolean invalidated) throws IOException, Exception
{
if(!invalidated)
{
server = new ServerSocket();
server.setReuseAddress(true);
server.bind(new InetSocketAddress(portnr));
portchanged = false;
}
else
{
//TODO: invalidate the old requestHandler
if(server != null)
{
try
{
server.close();
server = null;
}
catch (IOException ex)
{
Logger.getLogger(Controlserver.class.getName()).log(Level.SEVERE, null, ex);
}
}
if(server.isClosed())
{
System.out.println("closed biatch!");
}
else
{
System.out.println("surprise moddafakkaaaaa!!!");
}
//---------------------------------------------
//then make new requestHandler with new port
portchanged = true;
}
while(!portchanged)
{
if(server != null && !server.isClosed() && !invalidated)
{
Socket sock = server.accept();
System.out.println(server.getLocalPort());
System.out.println(sock.getLocalPort());
System.out.println("test");
handler = new Requesthandler(sock); //should be in a thread
handler.start();
System.out.println("ja harm");
System.out.println(server.getLocalPort());
System.out.println(sock.getLocalPort());
}
else
{
portchanged = true;
}
}
if(portchanged)
{
portchanged = false;
makeServerSocketWithPort(portnr, false);
}
}
再次,这通常很好。我可以浏览我的html页面。当我通过其中一个网页更改端口号时,它会在我的存储xml文件中正确存储和更改。 但当我更改我的套接字并通过该套接字立即导航到一个页面时,它表示它已关闭,并且在我重新启动我的应用程序之前无法正常工作。
我仍然在寻找绕过这次重启的方法。
嗯,我解开了这个谜。 事情是我只需要重新构建我的类,以更好地支持线程。我没有关闭套接字然后创建一个新线程,而是启动了一个新线程,然后关闭了套接字。经过一些摆弄后,它似乎工作正常。
感谢您的回复和内容。
答案 0 :(得分:2)
这是OS的正常服务器套接字行为。操作系统在WAIT_TIMEOUT状态下保持端口打开。要解决此问题,请尝试使用ServerSocket.setReuseAddress(boolean on)
。这将启用/禁用SO_REUSEADDR
套接字选项。文档Check here。
引用方法setReuseAddress
当TCP连接关闭时,连接可能会保持超时 在连接关闭后的一段时间内(通常是 称为TIME_WAIT状态或2MSL等待状态)。适用于应用 使用众所周知的套接字地址或端口可能无法实现 如果存在连接,则将套接字绑定到所需的SocketAddress 处于涉及套接字地址或端口的超时状态。
在使用绑定套接字之前启用SO_REUSEADDR bind(SocketAddress)允许套接字绑定,即使a 以前的连接处于超时状态。
答案 1 :(得分:2)
使用TCPview查看系统中所有已打开的端口。您可以关闭正在使用的端口。