这个问题的动机 我正在运行一个在非常昂贵的硬件上运行的巨大产品。出于测试目的而关闭它是不可能的,也不可能在生产环境中放置一个坏 jar。我需要尽可能确定,几乎可以确保我不会把生产环境搞得一团糟。
在我在登台设置上运行它之前,我需要检查下面的代码是否存在明显的问题(这同样昂贵)。
问题
我有一个基于套接字的应用程序,有时客户端不会显式发送 CloseConnection 请求。有时 IOException 不会发生,因为阻塞readObject
调用阻塞了线程。
我需要通过在超时后关闭连接来关闭这个线程。如果我从服务器收到新请求,则会刷新超时。
所以你会在下面看到 3 个部分
- 初始化
- while(true) 循环中的 readObject 调用,以及计划的服务重置
- 内流的实际关闭
代码
有人建议我使用 ScheduledExecutorService 而不是 Timer/TimerTask。
class StreamManager {
....
private ScheduledExecutorService activityTimeOut = Executors
.newSingleThreadScheduledExecutor();
private CloseConnectionOnTimeOut closeOnTimeOut = new CloseConnectionOnTimeOut();
....
public void initialize(Socket newClientSocket, ObjectInputStream newInputStream,
ObjectOutputStream newOutputStream, ThreadMonitor newThreadMonitor) {
....
closeOnTimeOut.setInputStream(myInputStream);
activityTimeOut.scheduleAtFixedRate(closeOnTimeOut, 0, Globals.INACTIVITY_TIME_OUT,
TimeUnit.MILLISECONDS);
}
public void run() {
....
while (true) {
try {
AMessageStrategy incomingCommand = (AMessageStrategy) myInputStream
.readObject();
activityTimeOut.shutdown();
activityTimeOut.scheduleAtFixedRate(closeOnTimeOut, 0,
Globals.INACTIVITY_TIME_OUT, TimeUnit.MILLISECONDS);
....
}
....
}
class CloseConnectionOnTimeOut implements Runnable {
private ObjectInputStream myInputStream;
public CloseConnectionOnTimeOut() {
}
public void setInputStream(ObjectInputStream myInputStream) {
this.myInputStream = myInputStream;
}
public void run() {
try {
myInputStream.close();
myOutputStream.close();
clientSocket.close();
log.info("Time out occured for client, closed connection forcefully.") ;
} catch (IOException e) {
e.printStackTrace();
log.fatal("Time out has occured, yet unable to clean up client connection. Keep a watch out on \"Size of clientStreamQ\"");
}
}
}
编辑:刚刚测试了一个较小的应用程序,它似乎工作。我仍然需要你的反馈。
再次编辑:
我根据建议修改了下面的代码。
初始化
private ScheduledExecutorService activityTimeOut = Executors
.newSingleThreadScheduledExecutor();
private Future<Void> timeoutTask ;
private CloseConnectionOnTimeOut closeOnTimeOut = new CloseConnectionOnTimeOut();
删除了此代码
closeOnTimeOut.setInputStream(myInputStream);
activityTimeOut.scheduleAtFixedRate(closeOnTimeOut, 0, Globals.INACTIVITY_TIME_OUT,
TimeUnit.MILLISECONDS);
在 readObject 前后替换
timeoutTask = (Future<Void>) activityTimeOut.scheduleAtFixedRate(
closeOnTimeOut.setInputStream(myInputStream), 0,
Globals.INACTIVITY_TIME_OUT, TimeUnit.MILLISECONDS);
AMessageStrategy incomingCommand = (AMessageStrategy) myInputStream
.readObject();
timeoutTask.cancel(true) ;
清理时
activityTimeOut.shutdown() ;