我已经实现了一个简单的文件上传下载机制.当用户单击文件名时,将使用以下HTTP标头下载该文件:
HTTP/1.1 200 OK Date: Tue, 30 Sep 2008 14:00:39 GMT Server: Microsoft-IIS/6.0 Content-Disposition: attachment; filename=filename.doc; Content-Type: application/octet-stream Content-Length: 10754
我也支持日文文件名.为此,我用这个java方法编码文件名:
private String encodeFileName(String name) throws Exception{ String agent = request.getHeader("USER-AGENT"); if(agent != null && agent.indexOf("MSIE") != -1){ // is IE StringBuffer res = new StringBuffer(); char[] chArr = name.toCharArray(); for(int j = 0; j < chArr.length; j++){ if(chArr[j] < 128){ // plain ASCII char if (chArr[j] == '.' && j != name.lastIndexOf(".")) res.append("%2E"); else res.append(chArr[j]); } else{ // non-ASCII char byte[] byteArr = name.substring(j, j + 1).getBytes("UTF8"); for(int i = 0; i < byteArr.length; i++){ // byte must be converted to unsigned int res.append("%").append(Integer.toHexString((byteArr[i]) & 0xFF)); } } } return res.toString(); } // Firefox/Mozilla return MimeUtility.encodeText(name, "UTF8", "B"); }
它到目前为止运作良好,直到有人发现它对长文件名不起作用.例如:???????????????2008.10.1?.doc
.如果我将其中一个单字节点更改为单字节下划线,或者如果我删除第一个字符,则它可以正常工作.即,它取决于点字符的长度和URL编码.以下是一些例子.
这是破碎的(???????????????2008.10.1?.doc
):
Content-Disposition: attachment; filename=%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%822008%2E10%2E1%e3%81%82.doc;
这是OK(???????????????2008_10.1?.doc
):
Content-Disposition: attachment; filename=%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%822008_10%2E1%e3%81%82.doc;
这也很好(???????????????2008.10.1?.doc
):
Content-Disposition: attachment; filename=%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%82%e3%81%822008%2E10%2E1%e3%81%82.doc;
有人有线索吗?
gmail处理文件名转义的方式有所不同:引用文件名(双引号),单字节句点不进行URL转义.这样,问题中的长文件名就可以了.
Content-Disposition: attachment; filename="%E3%81%82%E3%81%82%E3%81%82%E3%81%82%E3%81%82%E3%81%82%E3%81%82%E3%81%82%E3%81%82%E3%81%82%E3%81%82%E3%81%82%E3%81%82%E3%81%82%E3%81%822008.10.1%E3%81%82.doc"
但是,文件名的字节长度仍然存在限制(显然只有IE)(我假设是一个错误).因此,即使文件名仅由单字节字符组成,文件名的开头也会被截断.限制大约是160个字节.