问题
之前在开发过程中,遇到了一点问题,我要访问一个FTP服务器去下载文件详细情况如下:
1. 需要传入一个可能为中文的文件名;
2. 通过文件名去FTP上寻找该文件;
3. FTP服务器的命名编码为“GBK”;
思路
1.通过GET方法直接将文件名负载URL后面,但需要通过转码;
2.在Java Controller中收到参数后,进行解码,解码为正常数据;
3.用正常数据再转码为GBK,到Service中去调用FTP即可
4.(因公司安全考虑,我们需要在另一个模块中调用FTP)通过rest接口将文件名传出,另一模块获取到文件流转换为byte[]传回,调用response输出即可
总结
编码问题的解决方案:
Jquery对URL以及参数转码,据我所了解的主要应用encodeURI、encodeURIComponent,例如我需要传入变量名为fileDepence
var downloadDepence=fileID+"-"+filename;
window.location.href=encodeURI(ajaxurl+"/coadownload/downloadFile?fileDepence="+encodeURIComponent(downloadDepence));
这样我在后台就可以接收到转码过后的fileDepence这个串,通过验证encodeURIComponent会以“utf-8”进行转码,所以我们使用Java对其解码:
String viewItem=java.net.URLDecoder.decode(fileDepence, "utf-8");
这样得到的viewItem就与我们原本要传入的值一致了,如果传入的为中文文件名,则此时viewItem便是对应的中文文件名了。
之后我又了解一下,通过JS来完成GBK的转码比较麻烦,而采用Unicode的Java则比较方法,则同理,我们使用viewItem在以GBK来转一次码,就可以得到对应的FTP服务器中的文件名了。
String GBKItem= URLEncoder.encode(viewItem,"GBK");
FTP的解决方法
建立一个FTP连接的公用类
public static FtpClient connectFTP(String url, int port, String username, String password) {
//创建ftp
FtpClient ftp = null;
try {
//创建地址
SocketAddress addr = new InetSocketAddress(url, port);
//连接
ftp = FtpClient.create();
ftp.connect(addr);
//登陆
ftp.login(username, password.toCharArray());
ftp.setBinaryType();
} catch (FtpProtocolException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
logger.info("已登录到FTP服务器!" + new Date());
return ftp;
}
在另一个模块的Service中建立一个下载的方法,在rest接口访问的Controller中调用该方法。
public FileBtye downloadfilefromFTP(String fileDepence,HttpServletResponse response) throws IOException, FtpProtocolException {
FileBtye file=new FileBtye(); //FileBtye为一个只包含btye[]的对象,主要用于适合Controller的responsebody的json传出
response.setCharacterEncoding("utf-8");
fileDepence=fileDepence.replace("\"", ""); //rest接口传输过来的值需要做一定的格式化处理
FtpClient ftpClient= FTPManager.connectFTP(FTPHost, Integer.parseInt(FTPPort), FTPUserName, FTPPassWord); //通过properties文件读取
InputStream is = null;
ftpClient.changeDirectory("//filepath"); //文件目录
is = ftpClient.getFileStream(fileDepence); //获取文件流
ByteArrayOutputStream swapStream = new ByteArrayOutputStream();
byte[] buff = new byte[100]; //buff用于存放循环读取的临时数据
int rc = 0;
while ((rc = is.read(buff, 0, 100)) > 0) {
swapStream.write(buff, 0, rc);
}
byte[] fileByte = swapStream.toByteArray();
file.setFileinfo(fileByte);
is.close();
return file;
}
在本地Service中调用rest接口完成输出response输出即可,同时要注意response的设置
String viewItem=java.net.URLDecoder.decode(fileDepence, "utf-8");
//FTP服务器命名规则以GBK编码
String GBKItem= URLEncoder.encode(viewItem,"GBK");
//调用rest接口,拿到返回的FileBtye
FileBtye downloadStream=restTemplate.postForObject(gatewayRootUrl + GatewayUrlConstant.COA_DOWNLOAD + GatewayUrlConstant.DOWNLOAD_FILE, GBKItem, FileBtye.class);
if(downloadStream!=null){
//设置信息头
response.setCharacterEncoding("utf-8"); //设置编码
response.setContentType("multipart/form-data"); //根据文件确定文件格式
response.setHeader("Content-Disposition", "attachment;fileName=" + viewItem); //设置文件名
ServletOutputStream out;
//输出流输出
out = response.getOutputStream();
out.write(downloadStream.getFileinfo());
out.flush();
out.close();
logger.info("下载文件成功" + new Date());
}else {
logger.info("下载文件不存在" + new Date());
}