解决FTPClient下载网络文件线程挂起问题
今天在windows上调试FTP下载文件时,出险线程假死,代码如下:
if (inputStream != null) { byte[] data = null; ByteArrayOutputStream outStream = new ByteArrayOutputStream(); data = new byte[inputStream.available()]; int len = 0; while ((len = inputStream.read(data)) != -1) { outStream.write(data, 0, len); } data = outStream.toByteArray(); Base64.Encoder encoder = Base64.getEncoder(); re = encoder.encodeToString(data); }
通过打印日志排查到while死循环,查看read方法源码注释:
/* <p> If the length of <code>b</code> is zero, then no bytes are read and * <code>0</code> is returned; otherwise, there is an attempt to read at * least one byte. If no byte is available because the stream is at the * end of the file, the value <code>-1</code> is returned; otherwise, at * least one byte is read and stored into <code>b</code>.
*/
public int read(byte b[]) throws IOException {
return read(b, 0, b.length);
}
在date字节数组长度为1时,read方法返回0,while出现死循环,原因是因为网络传输数据不及时,导致inputStream.available()方法未取到本地内存的字节数据返回0。
由于下载的文件大小不是很大,默认给了一个2048大小的字节数组,于是做了以下优化:
if (inputStream != null) { byte[] data = null; ByteArrayOutputStream outStream = new ByteArrayOutputStream(); if (inputStream.available() == 0) { data = new byte[2048]; } else { data = new byte[inputStream.available()]; } int len = 0; while ((len = inputStream.read(data)) != -1) { outStream.write(data, 0, len); } data = outStream.toByteArray(); Base64.Encoder encoder = Base64.getEncoder(); re = encoder.encodeToString(data); }
再次调试,问题解决。
但是,在将程序部署到Linux服务器上依然假死,
问题分析:
FTPClient调用retrieveFileStream线程假死
FTPClient.listFiles()或者FTPClient.retrieveFile()方法时,就停止在那里,什么反应都没有,出现假死状 态。于是各种网上资料查找
FTP的两种传输模式:主动模式和被动模式
在调用这两个方法之前,调用 FTPClient.enterLocalPassiveMode();这个方法的意思就是每次数据连接之前,ftp client告诉ftp server开通一个端口来传输数据。因为ftp server可能每次开启不同的端口来传输数据,但是在linux上,由于防火墙安全限制,可能某些端口没有开启,所以就出现阻塞。
解决方法:ftpClient.enterLocalPassiveMode(); //开启本地被动模式,此代码设置在登陆之后或者之前都可以。
同样,如果上传的话需要enterRemotePassiveMode()//开启远程被动传输模式
感谢网络上的大神们的分享,不然又得钻牛角了。