Android中的HttpUrlConnection设置范围将被忽略

时间:2016-10-04 07:54:35

标签: java android httpurlconnection

我正在尝试使用Android从我的服务器获得206响应。

以下是代码。

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_splash);
    new AsyncTask<Void, Void, Void>() {
        @Override
        protected Void doInBackground(Void... params) {
            try {
                URL url = new URL("http://aviddapp.com/10mb.file");
                HttpURLConnection urlConnection = (HttpURLConnection) url.openConnection();
                urlConnection.setRequestProperty("Range", "bytes=1-2");
                urlConnection.connect();

                System.out.println("Response Code: " + urlConnection.getResponseCode());
                System.out.println("Content-Length: " + urlConnection.getContentLength());
                Map<String, List<String>> map = urlConnection.getHeaderFields();
                for (Map.Entry<String, List<String>> entry : map.entrySet()) {
                    System.out.println("Key : " + entry.getKey() +
                            " ,Value : " + entry.getValue());
                }
                InputStream inputStream = urlConnection.getInputStream();
                long size = 0;

                while(inputStream.read() != -1 )
                    size++;

                System.out.println("Downloaded Size: " + size);

            }catch(MalformedURLException mue) {
                mue.printStackTrace();
            }catch(IOException ioe) {
                ioe.printStackTrace();
            }
            return null;
        }
    }.execute();
}

这是输出:

I/System.out: Respnse Code: 200
I/System.out: Content-Length: -1
I/System.out: Key : null ,Value : [HTTP/1.1 200 OK]
I/System.out: Key : Accept-Ranges ,Value : [bytes]
I/System.out: Key : Cache-Control ,Value : [max-age=604800, public]
I/System.out: Key : Connection ,Value : [Keep-Alive]
I/System.out: Key : Date ,Value : [Tue, 04 Oct 2016 07:45:22 GMT]
I/System.out: Key : ETag ,Value : ["a00000-53e051f279680-gzip"]
I/System.out: Key : Expires ,Value : [Tue, 11 Oct 2016 07:45:22 GMT]
I/System.out: Key : Keep-Alive ,Value : [timeout=5, max=100]
I/System.out: Key : Last-Modified ,Value : [Tue, 04 Oct 2016 07:36:42 GMT]
I/System.out: Key : Server ,Value : [Apache/2.4.12 (Unix) OpenSSL/1.0.1e-fips mod_bwlimited/1.4]
I/System.out: Key : Transfer-Encoding ,Value : [chunked]
I/System.out: Key : Vary ,Value : [Accept-Encoding,User-Agent]
I/System.out: Key : X-Android-Received-Millis ,Value : [1475567127403]
I/System.out: Key : X-Android-Response-Source ,Value : [NETWORK 200]
I/System.out: Key : X-Android-Sent-Millis ,Value : [1475567127183]
I/System.out: Downloaded Size: 10485760

现在我做同样的事情就是纯粹的java。

public static void main(String... args) {
    try {
        URL url = new URL("http://aviddapp.com/10mb.file");
        HttpURLConnection urlConnection = (HttpURLConnection) url.openConnection();
        urlConnection.setRequestProperty("Range", "bytes=1-2");
        urlConnection.connect();

        System.out.println("Respnse Code: " + urlConnection.getResponseCode());
        System.out.println("Content-Length: " + urlConnection.getContentLength());
        Map<String, List<String>> map = urlConnection.getHeaderFields();
        for (Map.Entry<String, List<String>> entry : map.entrySet()) {
            System.out.println("Key : " + entry.getKey() +
                    " ,Value : " + entry.getValue());
        }
        InputStream inputStream = urlConnection.getInputStream();
        long size = 0;

        while(inputStream.read() != -1 )
            size++;

        System.out.println("Downloaded Size: " + size);

    }catch(MalformedURLException mue) {
        mue.printStackTrace();
    }catch(IOException ioe) {
        ioe.printStackTrace();
    }
}

这是输出

Respnse Code: 206
Content-Length: 2
Key : Keep-Alive ,Value : [timeout=5, max=100]
Key : null ,Value : [HTTP/1.1 206 Partial Content]
Key : Server ,Value : [Apache/2.4.12 (Unix) OpenSSL/1.0.1e-fips mod_bwlimited/1.4]
Key : Content-Range ,Value : [bytes 1-2/10485760]
Key : Connection ,Value : [Keep-Alive]
Key : Last-Modified ,Value : [Tue, 04 Oct 2016 07:36:42 GMT]
Key : Date ,Value : [Tue, 04 Oct 2016 07:42:17 GMT]
Key : Accept-Ranges ,Value : [bytes]
Key : Cache-Control ,Value : [max-age=604800, public]
Key : ETag ,Value : ["a00000-53e051f279680"]
Key : Vary ,Value : [Accept-Encoding,User-Agent]
Key : Expires ,Value : [Tue, 11 Oct 2016 07:42:17 GMT]
Key : Content-Length ,Value : [2]
Downloaded Size: 2

正如您所看到的,我在两种情况下都会得到不同的响应代码。似乎Android没有将Range传递给服务器?这里发生了什么?

PS:如果文件大小为1mb,我得到206.

4 个答案:

答案 0 :(得分:7)

我相对肯定错误不在Android代码中。

看起来服务器可能会产生虚假结果。

您可以使用third party tool (such as Postman)进行检查,以确定Web服务正在提供的标头。

我使用Postman的结果。如您所见,它正在提供HTTP 200(而不是206)。它也没有为上限Content-Length提供支持。如果服务器是您的,可能检查它是否配置正确。另请检查您的代码与其他服务器。

Postman results

答案 1 :(得分:5)

嗨,你可以尝试这段代码吗,看来我在这里得到HTTP 206

new Thread() {
            @Override
            public void run() {
                try {
                    URL url = new URL("http://aviddapp.com/10mb.file");
                    HttpURLConnection urlConnection = (HttpURLConnection) url.openConnection();
//                    urlConnection.setRequestMethod("HEAD");
                    urlConnection.setRequestProperty("Accept-Encoding", "");

                    urlConnection.setRequestProperty("Range", "bytes=1-2");
                    urlConnection.connect();

                    System.out.println("Response Code: " + urlConnection.getResponseCode());
                    System.out.println("Content-Length: " + urlConnection.getContentLength());
                    Map<String, List<String>> map = urlConnection.getHeaderFields();
                    for (Map.Entry<String, List<String>> entry : map.entrySet()) {
                        System.out.println("Key : " + entry.getKey() +
                                " ,Value : " + entry.getValue());
                    }
                    InputStream inputStream = urlConnection.getInputStream();
                    long size = 0;

                    while (inputStream.read() != -1)
                        size++;

                    System.out.println("Downloaded Size: " + size);

                } catch (IOException ioe) {
                    ioe.printStackTrace();
                }
            }
        }.start();

您只需在异步任务中添加它。

检查此结果是否在此行上注释。

urlConnection.setRequestProperty("Accept-Encoding", "");

答案 2 :(得分:3)

我不确定这是否与默认情况下由android实现清除的Content-Length有关。请查看源代码中的以下代码段,其中默认情况下会执行gzip压缩。

https://android.googlesource.com/platform/libcore/+/2e317a02b5a8f9b319488ab9311521e8b4f87a0a/luni/src/main/java/java/net/HttpURLConnection.java

  

默认情况下,HttpURLConnection的此实现   请求服务器自动使用gzip压缩   解压缩getInputStream()的呼叫者的数据。该   清除Content-Encoding和Content-Length响应头   这种情况。

您可以尝试使用以下代码禁用默认缓存来检查它是否有效吗?

urlConnection.setRequestProperty("Accept-Encoding", "identity");

PS:抱歉格式不正确。使用移动版SO。

答案 3 :(得分:1)

你是否尝试过这样:

    urlConnection.setRequestProperty("Range", "bytes=1000-");