我正在使用commons-net的Java FTPClient类从ftp服务器下载大文件。
如果我检测到某些连接失败,我希望文件从某个偏移量重新开始下载:我的客户端连接断开或我的服务器关闭连接。
使用client.retrieveFile(filename, out)
如果我在下载文件时关闭客户端连接,则不会出现任何异常,程序只是冻结而不下载或执行任何操作。我该如何触发此事件?我看到了一个FTPConnectionClosedException,但它只在服务器关闭我的连接时被调用。
答案 0 :(得分:0)
您可以保留某种序列ID,它可以表示已下载了多少数据,并在合法完成后将其设置为某个特殊值。
sequence=0;
try{
while(notDone){
n = client.retrieveFile(filename, out); //bytes read
sequence =+ n;
}finally{
if(sequence != SPECIAL_VALUE){
throw new customException();
}
}
最终,您的代码将执行finally块,您可以决定如何处理它。或者,如果您想要更多fail fast
解决方案,可以通过衡量上次更新sequence
的时间来使用超时。
希望这有帮助!
答案 1 :(得分:0)
在下载进行时,您可以定期致电Para ftpClient.getState()
。如果连接断开,则抛出SocketTimeoutException。另外,以前您需要通过ftp.setControlKeepAliveTimeout(5)
这是示例:
ftpClient.setControlKeepAliveTimeout(5);
Callable<Boolean> downloadTask = () -> {
try {
return ftpClient.retrieveFile(name, counterOutput);
} catch (IOException e) {
Logger.error(e.getMessage(), e);
return false;
}
};
Callable<Boolean> connectionCheckTask = () -> {
try {
while(true) {
Thread.sleep(5000);
ftpClient.getStatus();
}
} catch (SocketTimeoutException e) {
Logger.error("Connection is broken", e);
return false;
} catch (InterruptedException e) {
Logger.error(ex.getMessage, e);
return false;
}
};
ExecutorService execSvc = Executors.newCachedThreadPool();
Boolean result = execSvc.invokeAny(Arrays.asList(downloadTask, connectionCheckTask));
execSvc.shutdown();
我认为这不是完美的解决方案,但对我有用。