0

我使用三个 servlet 来提供文件以供下载:

  • ByteArrayDownloadServlet:用于小文件,例如报告或数据库中的文件
  • FileDownloadServlet:用于小到大的文件
  • MultipleFileDownloadServlet:使用请求的文件创建一个 zip 并将其流式传输

它们基于以下实现: 链接文本

我收到了几起关于下载损坏的投诉。问题是我无法模拟或在错误中找到模式:

  • 有时有大文件
  • 有时当用户请求下载多个文件和一个 zip 文件并动态创建时
  • 有时文件较小,但许多用户同时请求

在帖子的上述评论中,有人报告了类似的问题,但没有解决方案。我也从这里读了很多线程,我越接近: 链接文本

有没有人遇到过类似的问题或有一些有效的示例代码?

谢谢,费利佩

@Override
@SuppressWarnings("unchecked")    
protected void doGet(HttpServletRequest request, HttpServletResponse response)
    throws ServletException, IOException
{
    HttpSession session = request.getSession();
    List<File> selectedFileList = (List<File>) session.getAttribute("selectedFileList");

    if(selectedFileList == null)
    {
        response.sendError(HttpServletResponse.SC_PRECONDITION_FAILED, "Lista de arquivos não informada");
        return;
    }

    response.reset();
    response.setContentType("application/zip");        

    response.setHeader("Content-Disposition", "attachment; filename=\""
        + "atualizacoes_"
        + new Date().getTime() + ".zip" + "\"");

    ZipOutputStream output = null;

    try
    {
        output = new ZipOutputStream(response.getOutputStream());

        for(File file : selectedFileList)
        {
            InputStream input = new FileInputStream(file);
            output.putNextEntry(new ZipEntry(file.getName()));                

            byte[] buffer = new byte[DownloadHandler.DEFAULT_BUFFER_SIZE];
            int length;
            while((length = input.read(buffer)) > 0)
            {
                output.write(buffer, 0, length);
            }

            output.closeEntry();
            input.close();
        }            

     output.finish();
     output.flush();
     output.close();
  }
  catch(Exception e) 
  {
      if(!(e instanceof ClientAbortException))
      {
          new ExceptionMail(getClass().getSimpleName(), e);
      }
    }
  finally
  {            
        session.removeAttribute("selectedFileList");        
  }
4

3 回答 3

2

从 servlet 随机损坏下载的最常见原因是 servlet 不是线程安全的和/或它正在将字节作为字符读取。在同一会话或 servlet 上下文中的请求之间共享请求或基于会话的数据也是导致此问题的可能原因。

于 2010-10-06T04:14:38.837 回答
0

您不应关闭输出流,因为它由 servlet 容器管理。我不确定冲洗。

于 2012-12-24T07:20:21.567 回答
0

您的代码在下面几行中有一个严重的流程。

int length;
        while((length = input.read(buffer)) > 0)
        {
            output.write(buffer, 0, length);
        }

您的“输入”是 FileInputStream 对吗?如何确保 FileInputStream 在整个迭代过程中始终有超过 0 个字节可用?相反,它必须写如下。

int length;
        while((length = input.read(buffer)) != -1)
        {
            output.write(buffer, 0, length);
        }
于 2013-09-02T15:05:22.180 回答