【问题标题】:Java: ZIP Files downloaded with HTTPUrlConnection are corruptedJava:使用 HTTPUrlConnection 下载的 ZIP 文件已损坏
【发布时间】:2019-03-24 12:18:53
【问题描述】:

我正在将一个 webapp 从 Java 8 迁移到 11(以及 Tomcat 8 到 9),并且我有一个使用以下方法从服务下载 ZIP 存档文件的客户端:

public HTTPResponse doGet(String aUrl, HashMap<String,String> aRequestParams, HashMap<String,String> aRequestProperties)
throws Exception
{
    try
    {
        String lUrl = aUrl;
        if (aRequestParams != null && aRequestParams.size() > 0)
        {
             StringBuffer lBodyStringBuffer = new StringBuffer();
             for(String lParam : aRequestParams.keySet())
             {
                 String lValue = aRequestParams.get(lParam);
                 if(lValue != null && !"".equals(lValue.trim()))
                 {
                     if(lBodyStringBuffer.length() > 0)
                     {
                         lBodyStringBuffer.append("&");
                     }
                     lBodyStringBuffer.append(URLEncoder.encode(lParam, sDEFAULTENCODING)).append("=").append(URLEncoder.encode(lValue, sDEFAULTENCODING));
                 }
             }
             String lParamString = lBodyStringBuffer.toString();
             if (lParamString != null && lParamString.length() > 0)
             {
                 if (!(lUrl.endsWith(sURLPARAMSLEADER) || aUrl.endsWith(sURLPARAMSSEPARATOR)))
                 {
                     if (lUrl.indexOf(sURLPARAMSLEADER) > -1)
                     {
                         lUrl = lUrl + sURLPARAMSSEPARATOR;
                     }
                     else
                     {
                         lUrl = lUrl + sURLPARAMSLEADER;
                     }
                 }
                 lUrl = lUrl + lParamString;
             }

        }
        HttpURLConnection lConnection = createConnection(lUrl,sREQUESTETHOD_GET,null, aRequestProperties);
        HTTPResponse lReturn = getResponseFromConnection(lConnection);
        return lReturn;
    }
    catch(Exception lException)
    {
        throw new Exception("Fehler beim Durchführen der Anfrage: " + lException.getMessage(), lException);
    }
}

private HTTPResponse getResponseFromConnection(HttpURLConnection aConnection)
throws Exception
{
    InputStream lConnectionInputStream = null;
    ByteArrayOutputStream lResponseByteArrayOutputStream = null;
    try
    {
        aConnection.setRequestProperty("Accept", "application/zip");
        int lStatusCode = aConnection.getResponseCode();
        String lResponseCharset = getCharsetFromResponseContentType(aConnection.getContentType());
        if (lResponseCharset == null)
        {
            if (lResponseCharset == null ||lResponseCharset.trim().length() == 0)
            {
                lResponseCharset = "UTF-8";
            }
        }
        if (HttpURLConnection.HTTP_OK == lStatusCode)
        {
            lConnectionInputStream = aConnection.getInputStream();
        }
        else
        {
            lConnectionInputStream = aConnection.getErrorStream();
        }
        String lMessage = "";
        if (lConnectionInputStream != null)
        {
            lResponseByteArrayOutputStream = new ByteArrayOutputStream();
            int lBufferSize = 4096;
            byte[]  lBuffer = new byte[lBufferSize];
            int lLength = 0;
            while ((lLength = lConnectionInputStream.read(lBuffer, 0, lBufferSize)) != -1)
            {
                lResponseByteArrayOutputStream.write(lBuffer, 0, lLength);
            }
            byte[] lResponseByte =  lResponseByteArrayOutputStream.toByteArray();
            lMessage = new String (lResponseByte,lResponseCharset);
        }
        HTTPResponse lReturn = new HTTPResponse(lStatusCode, lMessage);
        return lReturn;
    }
    catch(Exception lException)
    {
        throw lException;
    }
    finally
    {
        if (lResponseByteArrayOutputStream != null)
        {
            try{lResponseByteArrayOutputStream.close();}catch(Exception e){}
        }
        if (lConnectionInputStream != null)
        {
            try{lConnectionInputStream.close();}catch(Exception e){}
        }
    }
}

这就是我实际通过 HTTP Get 调用服务并保存数据的方式:

HTTPResponse lResponse = new HTTPRequest().doGet("http://localService.com/, null, null);
FileOutputStream lFileOutputStream = new FileOutputStream("exampleFile.zip", false);
lFileOutputStream.write(lResponse.getMessage().getBytes());
lFileOutputStream.close();

所以这以前可以工作,我对可能发生的变化感到头疼。当我在浏览器中下载带有 url 的 ZIP 文件时,一切似乎都很好,因此服务似乎可以工作。但是对于我的客户,ZIP 文件已损坏且无法打开。它们不是空的,但它们的大小不同:令人惊讶的是,损坏的文件比通过浏览器下载的文件大 50%。

有谁知道这里的问题是什么?

【问题讨论】:

  • 压缩文件不包含字符。它包含原始字节。将字节转换为字符串,然后将字符串转换回字节是错误的。
  • JB Nizet 所说的。 Java 不是 C。String 不是字节容器。将字节转换为字符串并返回字节损坏您的字节。
  • lResponse.getMessage()的返回类型是什么?它是 java.lang.String 吗?如果是,那就是问题所在(正如 JB Nizet 所说)。
  • 顺便说一下,您可以将 URL 保存到仅使用 try (InputStream stream = aConnection.getInputStream()) { Files.copy(stream, Paths.get("exampleFile.zip")); } 的文件中。由于您使用的是 Java 11,因此您可能需要查看 java.net.http package 以执行 HTTP 请求。

标签: java http httpurlconnection tomcat9 java-11


【解决方案1】:

好的,问题是方法getResponseFromConnection 的返回类型String。我改了,现在直接用VGR建议的File.copy()方法写文件。

【讨论】:

    猜你喜欢
    • 2014-08-09
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2019-06-22
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多