首先在文件下载时,请求类型需要设置编码 :
request.setCharacterEncoding("UTF-8"); response.setContentType("application/octet-stream; charset=utf-8"); response.setHeader("Content-disposition", "attachment;" + UserAgentUtil.encodeFileName(request, fileName)); response.setHeader("Content-Length", String.valueOf(fileLength));
最终文件名处理交给UserAgentUtil的encodeFileName:
/** * 获取客户端浏览器类型、编码下载文件名 * * @param request * @param fileName * @return String */ public static String encodeFileName(HttpServletRequest request, String fileName) { String userAgent = request.getHeader("User-Agent"); String rtn = ""; try { String new_filename = URLEncoder.encode(fileName, "UTF8"); // 如果没有UA,则默认使用IE的方式进行编码,因为毕竟IE还是占多数的 rtn = "filename=\"" + new_filename + "\""; if (userAgent != null) { userAgent = userAgent.toLowerCase(); // IE浏览器,只能采用URLEncoder编码 if (userAgent.indexOf("msie") != -1) { rtn = "filename=\"" + new_filename + "\""; } // Opera浏览器只能采用filename* else if (userAgent.indexOf("opera") != -1) { rtn = "filename*=UTF-8''" + new_filename; } // Safari浏览器,只能采用ISO编码的中文输出 else if (userAgent.indexOf("safari") != -1) { rtn = "filename=\"" + new String(fileName.getBytes("UTF-8"), "ISO8859-1") + "\""; } // Chrome浏览器,只能采用MimeUtility编码或ISO编码的中文输出 else if (userAgent.indexOf("applewebkit") != -1) { new_filename = MimeUtility.encodeText(fileName, "UTF8", "B"); rtn = "filename=\"" + new_filename + "\""; } // FireFox浏览器,可以使用MimeUtility或filename*或ISO编码的中文输出 else if (userAgent.indexOf("mozilla") != -1) { rtn = "filename*=UTF-8''" + new_filename; } } } catch (UnsupportedEncodingException e) { e.printStackTrace(); } return rtn; }
这样在进行文件下载时,使用这个工具类对文件名进行一次编码,就可以处理不同浏览器下载文件乱码的问题了。
参考 开源中国