gpt4 book ai didi

Java客户端/服务器关闭后资源泄漏

转载 作者:行者123 更新时间:2023-12-02 11:32:51 25 4
gpt4 key购买 nike

我正在尝试实现一个java命令行聊天服务器和客户端。我使用 4 个类,Server.java 设置一个 ServerSocket 并接受所有连接。 Client.java 可以连接到服务器,然后使用 SendMessages.java 和 ReceiveMessages.java 创建客户端和服务器的 2 个线程。线程负责从 stdin 获取输入并将其发送到套接字的输出流,并获取传入的输入流将其打印到 stdout。启动服务器并连接客户端时一切正常,聊天也正常。然而,当客户端或服务器终止时,就会导致资源泄漏。我希望SendMessages类和ReceiveMessages类能够检测客户端或服务器的连接何时终止并关闭所有资源以避免资源泄漏。这是代码:

Server.java

import java.net.*;
import java.io.*;
import java.util.*;
public class Server{
public static void main(String[] args) throws SocketException,IOException{
ServerSocket s = new ServerSocket(8000);
s.setSoTimeout(10000000);
while(true){
Socket clientSocket = s.accept();
handle(clientSocket);

}
}

static void handle(Socket clientSocket) throws IOException{
System.out.println("connection accepted from " + clientSocket.getRemoteSocketAddress());
receiveMessages(clientSocket);
sendMessages(clientSocket);
}

static void receiveMessages(Socket clientSocket) throws IOException{
(new Thread(new ReceiveMessages(clientSocket))).start();;

}

static void sendMessages(Socket clientSocket)throws IOException{
(new Thread(new SendMessages(clientSocket))).start();
}
}

Client.java

import java.net.*;
import java.io.*;
import java.util.*;

public class Client{


public static void main(String[] args) throws IOException, UnknownHostException, ConnectException{
String hostname = args[0];
int port = Integer.parseInt(args[1]);
Socket s = null;
s = connect(hostname, port);
handle(s);
}


public static Socket connect(String hostname, int port) throws UnknownHostException, IOException, ConnectException{
Socket s = null;
try{
s = new Socket(hostname, port);
} catch(ConnectException e){
System.out.println("Connect Exception caught!");
}
return s;
}

static void handle(Socket clientSocket) throws IOException, UnknownHostException{
receiveMessages(clientSocket);
sendMessages(clientSocket);
}

static void receiveMessages(Socket clientSocket) throws IOException, NullPointerException{
(new Thread(new ReceiveMessages(clientSocket))).start();
}

static void sendMessages(Socket clientSocket)throws IOException, NullPointerException{
(new Thread(new SendMessages(clientSocket))).start();
}

}

SendMessages.java

import java.io.*;
import java.net.*;
import java.util.*;
public class SendMessages implements Runnable{
Socket clientSocket;

public SendMessages(Socket clientSocket){
this.clientSocket = clientSocket;
}

public void run(){
System.out.println("SendMessages thread has started.");
Scanner sc = new Scanner(System.in);
PrintWriter out = null;
try{
out = new PrintWriter(clientSocket.getOutputStream(), true);
}
catch(IOException | NullPointerException e){}

String message;

while(true){
message = sc.nextLine();
out.println(message);
if(out.checkError()){
out.close();
sc.close();
System.out.println("SendMessages closed");
}
}

}
}

ReceiveMessages.java

import java.io.*;
import java.net.*;
import java.util.*;
public class ReceiveMessages implements Runnable{
Socket clientSocket;
public ReceiveMessages(Socket clientSocket){
this.clientSocket = clientSocket;
}
public void run(){
System.out.println("ReceiveMessages thread has started.");
BufferedReader in = null;
try{
in = new BufferedReader(new InputStreamReader(clientSocket.getInputStream()));
}catch(IOException|NullPointerException e){}
String message;
try{
while(true){
while((message = in.readLine())!= null){
System.out.println(message);
}

}
} catch(IOException|NullPointerException e){
System.out.println("ReceiveMessages resources closed.");
try{
in.close();
clientSocket.close();
}
catch(IOException f){}
}
}
}

谢谢!

最佳答案

所以我最近遇到了这样的问题。

问题是,通过关闭连接的一侧,"half-open"连接仍然存在(不再是真正的连接)。

Socket API 没有选项来检查合作服务/设备是否仍然存在。

那么该怎么办呢?我自己更喜欢心跳的概念。每 10 秒(或任何其他时间范围,由您决定)就会向套接字的输出流发送一次心跳。当流不再可用时,此操作会引发 IOException。因此,您可以捕获异常并处理 catch block 中的所有“关闭”操作。

示例:

    public class Heartbeater
implements Runnable
{
private OutputStream os;
private ServerHandler servhand;
private Logger logger = Logger.getLogger( Logger.GLOBAL_LOGGER_NAME );

//NOTE: ServerHandler is my class which handles the socket of the server after ServerSocket.accept();
public Heartbeater( OutputStream os, ServerHandler servhand )
{
this.servhand = servhand;
this.os = os;
}

@Override
public void run()
{
try
{
Thread.currentThread().setName( "Heartbeater" );
// while the handler's connection is alive
while ( servhand.isConnectionAlive() )
{

Thread.sleep( 10000 );

// dummy write to trigger the exception if the client does not respond properly
os.write( "_HEARTBEAT_".getBytes() );

}
}
catch ( InterruptedException e )
{
logger.log( Level.SEVERE, "HEARTBEATER GOT INTERRUPTED", e );
Thread.currentThread().interrupt();
}
catch ( IOException e )
{
logger.log( Level.INFO, "The connection to the client has been lost." );
// Here you would define the resource close operation.
servhand.setConnectionAlive( false );
}
}
}

关于Java客户端/服务器关闭后资源泄漏,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49173092/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com