2

我正在使用 servlet,它用于打开文档,如 doc、txt、pdf、ppt 等。

我的代码片段如下。

Documents document = db.getDocument(docCode);
 String contentType = document.getDocMimeType();
 byte[] docContentBytes = document.getDocContentBytes();  

 ServletOutputStream out = response.getOutputStream ();
 response.setHeader("X-UA-Compatible", "IE=8");
 response.setHeader("Content-disposition", "attachment;filename=\"Document\""); 
 response.setHeader("Pragma","private");
 response.setHeader("Cache-Control","must-revalidate, post-check=0, pre-check=0");
 response.setHeader("Content-Transfer-Encoding","binary");

 if(contentType!=null){
     response.setContentType(contentType);
 }else{
     response.setContentType("application/pdf");
 }

 BufferedInputStream bis = null;
 BufferedOutputStream bos = null;
 ByteArrayInputStream bais = null;

 if(docContentBytes != null) {
  try{
         bais = new ByteArrayInputStream(docContentBytes);
         bis = new BufferedInputStream(bais);


         bos = new BufferedOutputStream(out);
         byte[] buff = new byte[2048];
         int bytesRead;
         // Simple read/write loop.
         while(-1 != (bytesRead = bis.read(buff, 0, buff.length))) {
         bos.write(buff, 0, bytesRead);
         }
 }catch(final MalformedURLException e) {
 System.out.println ( "MalformedURLException." );
 throw e;
 } catch(final IOException e) {
 System.out.println ( "IOException." );
 throw e;
 } finally {
     if (bais != null)
         bais.close();

         if (bis != null)
             bis.close();

         if (bos != null)
             bos.close();
 }
 } 

现在,当我尝试打开多个文档时,一段时间后我会从 tomcat 服务器收到损坏的管道错误。

我的 DataSource 实现如下。

<Resource name="jdbc/TEST_DS"
                        auth="Container"
                        type="javax.sql.DataSource"
                        driverClassName="com.microsoft.sqlserver.jdbc.SQLServerDriver"
                        url="jdbc:sqlserver://hostName;databaseName=TEST"
                        username="test"
                        password="testPwd"
                        maxPoolSize="50" 
                removeAbandoned="true"
                        removeAbandonedTimeout="1000"
                        logAbandoned="true"
                        />

任何人都可以建议我在这段代码中需要修改什么?

4

4 回答 4

1

出现该错误有几个原因:

  • 服务器尝试通信时网络中断。
  • 用户取消请求。
  • 数据库已关闭或拒绝连接。

您需要做的是按以下顺序关闭:

finally {
      if(bos != null)
         bos.close();
      if(bis != null)
         bis.close();
      if(bais != null)
         bais.close();
}

希望这可以帮助!

于 2013-10-28T20:21:37.270 回答
1

您已经在 byte[] docContentBytes 内存中加载了所有文件内容,那么为什么需要缓冲呢?

但是,如果异常是由 tomcat 和浏览器/客户端之间的通信错误引起的,请尝试删除缓冲并设置内容长度。

否则,如果异常是由 tomcat 和 SQL 服务器之间的通信错误引起的,则您的问题出在以下之一:

  • db类实现
  • Documents执行
  • Microsoft SQL Server 安装中的设置/调整

然而,这就是我编写这个 servlet 的方式:

@Override
public void doPost(HttpServletRequest request, HttpServletResponse response) throws IOException
{
    Database db = Database.getInstance();

    String docCode = request.getParameter("docCode");
    if(docCode == null || docCode.isEmpty()) throw new IllegalArgumentException("docCode is null");

    Documents document = db.getDocument(docCode);
    if(document == null) throw new IllegalStateException("invalid docCode: " + docCode);

    byte[] docContentBytes = document.getDocContentBytes();
    if(docContentBytes == null) throw new IllegalStateException("document " + docCode + " has no content");

    String contentType = document.getDocMimeType();
    if(contentType == null) contentType = "application/octet-stream";

    response.setHeader("X-UA-Compatible", "IE=8");
    response.setHeader("Content-disposition", "attachment;filename=\"Document\"");
    response.setHeader("Pragma", "private");
    response.setHeader("Cache-Control", "must-revalidate, post-check=0, pre-check=0");
    response.setHeader("Content-Transfer-Encoding", "binary");
    response.setContentType(contentType);
    response.setContentLength(docContentBytes.length);

    response.getOutputStream().write(docContentBytes);
}
于 2013-11-03T11:10:22.970 回答
1

您不应该关闭响应的输出流。我不确定这是否会导致您的管道损坏错误,但我肯定会尝试一下。

Documents document = db.getDocument(docCode);
String contentType = document.getDocMimeType();
byte[] docContentBytes = document.getDocContentBytes();

ServletOutputStream out = response.getOutputStream();
response.setHeader("X-UA-Compatible", "IE=8");
response.setHeader("Content-disposition", "attachment;filename=\"Document\"");
response.setHeader("Pragma", "private");
response.setHeader("Cache-Control", "must-revalidate, post-check=0, pre-check=0");
response.setHeader("Content-Transfer-Encoding", "binary");

if (contentType != null) {
    response.setContentType(contentType);
} else {
    response.setContentType("application/pdf");
}


if (docContentBytes != null) {
    try {
        ByteArrayInputStream bais = new ByteArrayInputStream(docContentBytes);

        byte[] buff = new byte[2048];

        int bytesRead;
        while (-1 != (bytesRead = bais.read(buff))) {
            out.write(buff, 0, bytesRead);
        }
    } finally {
        out.flush();
    }
}

以下是其他一些指针:

  • 缓冲字节数组输入流没有意义;你已经有内存中的数据

  • 关闭缓冲流也会关闭底层流

  • 关闭字节数组流不会做任何事情

  • 缓冲响应的输出流没有意义,因为您已经在代码中使用了缓冲机制

  • 在处理流/文件(例如IOUtils)时,第三方库(例如 common-io)非常有用

  • 发布您的异常堆栈跟踪的(一部分)您正在发布有关异常的问题

于 2013-11-02T17:48:33.127 回答
1

这里的答案来看,似乎关闭的顺序可能会导致问题。

更改此代码...

finally {
     if (bais != null)
         bais.close();

         if (bis != null)
             bis.close();

         if (bos != null)
             bos.close();
 }

至...

finally {
         bais.close();
         bos.close();
         bis.close();
 }
于 2013-10-28T11:29:45.463 回答