【问题标题】:Resume http file download in java在java中恢复http文件下载
【发布时间】:2011-06-04 13:45:21
【问题描述】:
URL url = new URL("http://download.thinkbroadband.com/20MB.zip");

URLConnection connection = url.openConnection();
File fileThatExists = new File(path); 
OutputStream output = new FileOutputStream(path, true);
connection.setRequestProperty("Range", "bytes=" + fileThatExists.length() + "-");

connection.connect();

int lenghtOfFile = connection.getContentLength();

InputStream input = new BufferedInputStream(url.openStream());
byte data[] = new byte[1024];

long total = 0;

while ((count = input.read(data)) != -1) {
    total += count;

    output.write(data, 0 , count);
}

在这段代码中,我尝试恢复下载。目标文件为 20MB。但是当我在 10MB 上停止下载,然后继续,我得到文件大小为 30MB 的文件。似乎它继续写入文件,但无法从服务器部分下载。 Wget -c 非常适合这个文件。如何恢复文件下载?

【问题讨论】:

标签: java android http download


【解决方案1】:
 HttpURLConnection connection = (HttpURLConnection) url.openConnection();
    if(ISSUE_DOWNLOAD_STATUS.intValue()==ECMConstant.ECM_DOWNLOADING){
        File file=new File(DESTINATION_PATH);
        if(file.exists()){
             downloaded = (int) file.length();
             connection.setRequestProperty("Range", "bytes="+(file.length())+"-");
        }
    }else{
        connection.setRequestProperty("Range", "bytes=" + downloaded + "-");
    }
    connection.setDoInput(true);
    connection.setDoOutput(true);
    progressBar.setMax(connection.getContentLength());
     in = new BufferedInputStream(connection.getInputStream());
     fos=(downloaded==0)? new FileOutputStream(DESTINATION_PATH): new FileOutputStream(DESTINATION_PATH,true);
     bout = new BufferedOutputStream(fos, 1024);
    byte[] data = new byte[1024];
    int x = 0;
    while ((x = in.read(data, 0, 1024)) >= 0) {
        bout.write(data, 0, x);
         downloaded += x;
         progressBar.setProgress(downloaded);
    }

这不是我的代码,但它可以工作。

【讨论】:

  • 你好,'ISSUE_DOWNLOAD_STATUS.intValue()'我不明白这个值,请给我这个参考。
  • 我预计此代码将无法正常工作。如果 if(ISSUE_DOWNLOAD_STATUS.intValue()==ECMConstant.ECM_DOWNLOADING) 的条件失败,则会在未定义“已下载”变量的情况下执行 else{} 语句。
  • @asiby 假设我们都可以编写一点代码,这段代码给你一个提示,告诉你如何解决这个问题,而不是完全解决。
  • 嘿@Ceetn,如果您不能提供任何建设性的帮助,请压缩它。这里的大多数人都在努力提供功能性解决方案。这个看起来像是来自它可能工作的某个地方,但是缺少变量声明。我尝试使用它,我几乎花了几分钟试图理解它。什么是 ISSUE_DOWNLOAD_STATUS 和 ECMConstant?你可能很清楚。但是你认为人们在这里寻求帮助时可以编码是完全错误的。如果你的老师在你的编程课上表现得那样,你会有不同的反应。
  • 我不明白这个答案是如何获得更多支持并被接受的。很多没有解释的东西,外部参考。常量没有价值。在 if 分支中设置的下载属性的值,在 else 分支中使用。 if 和 else 分支是一样的。为什么 setDoOutput 设置为 true?任何 POST 都通过了吗?在 3 个地方使用相同的缓冲区大小,没有使用常量。根本没有错误处理。
【解决方案2】:

我猜你面临的问题是在url.openConnection() 之后调用url.openStream()

url.openStream() 等价于url.openConnection().getInputStream()。因此,您请求该 url 两次。特别是第二次,它没有指定范围属性。因此下载总是从头开始。

您应该将url.openStream() 替换为connection.getInputStream()

【讨论】:

  • 非常感谢。你真的救了我!我遇到了类似的问题,您的解决方案很完美。谢谢!
  • 我花了大约一个小时来寻找我的问题,您先生,您让我开心并解决了它!非常感谢!
【解决方案3】:

这就是我用来以块的形式下载文件的内容 使用进度更新 UI。

/*
 * @param callback = To update the UI with appropriate action
 * @param fileName = Name of the file by which downloaded file will be saved.
 * @param downloadURL = File downloading URL
 * @param filePath = Path where file will be saved
 * @param object = Any object you want in return after download is completed to do certain operations like insert in DB or show toast
 */

public void startDownload(final IDownloadCallback callback, String fileName, String downloadURL, String filePath, Object object) {
    callback.onPreExecute(); // Callback to tell that the downloading is going to start
    int count = 0;
    File outputFile = null; // Path where file will be downloaded
    try {
        File file = new File(filePath);
        file.mkdirs();
        long range = 0;
        outputFile = new File(file, fileName);
        /**
         * Check whether the file exists or not
         * If file doesn't exists then create the new file and range will be zero.
         * But if file exists then get the length of file which will be the starting range,
         * from where the file will be downloaded
         */
        if (!outputFile.exists()) {
            outputFile.createNewFile();
            range = 0;
        } else {
            range = outputFile.length();
        }
        //Open the Connection
        URL url = new URL(downloadURL);
        URLConnection con = url.openConnection();
        // Set the range parameter in header and give the range from where you want to start the downloading
        con.setRequestProperty("Range", "bytes=" + range + "-");
        /**
         * The total length of file will be the total content length given by the server + range.
         * Example: Suppose you have a file whose size is 1MB and you had already downloaded 500KB of it.
         * Then you will pass in Header as "Range":"bytes=500000".
         * Now the con.getContentLength() will be 500KB and range will be 500KB.
         * So by adding the two you will get the total length of file which will be 1 MB
         */
        final long lenghtOfFile = (int) con.getContentLength() + range;

        FileOutputStream fileOutputStream = new FileOutputStream(outputFile, true);
        InputStream inputStream = con.getInputStream();

        byte[] buffer = new byte[1024];

        long total = range;
        /**
         * Download the save the content into file
         */
        while ((count = inputStream.read(buffer)) != -1) {
            total += count;
            int progress = (int) (total * 100 / lenghtOfFile);
            EntityDownloadProgress entityDownloadProgress = new EntityDownloadProgress();
            entityDownloadProgress.setProgress(progress);
            entityDownloadProgress.setDownloadedSize(total);
            entityDownloadProgress.setFileSize(lenghtOfFile);
            callback.showProgress(entityDownloadProgress);
            fileOutputStream.write(buffer, 0, count);
        }
        //Close the outputstream
        fileOutputStream.close();
        // Disconnect the Connection
        if (con instanceof HttpsURLConnection) {
            ((HttpsURLConnection) con).disconnect();
        } else if (con instanceof HttpURLConnection) {
            ((HttpURLConnection) con).disconnect();
        }
        inputStream.close();
        /**
         * If file size is equal then return callback as success with downlaoded filepath and the object
         * else return failure
         */
        if (lenghtOfFile == outputFile.length()) {
            callback.onSuccess(outputFile.getAbsolutePath(), object);
        } else {
            callback.onFailure(object);
        }
    } catch (Exception e) {
        e.printStackTrace();
        callback.onFailure(object);
    }
}

接口 IDownloadCallback {

    void onPreExecute(); // Callback to tell that the downloading is going to start
    void onFailure(Object o); // Failed to download file
    void onSuccess(String path, Object o); // Downloaded file successfully with downloaded path
    void showProgress(EntityDownloadProgress entityDownloadProgress); // Show progress
}

公共类 EntityDownloadProgress {

    int progress; // range from 1-100
    long fileSize;// Total size of file to be downlaoded
    long downloadedSize; // Size of the downlaoded file

    public void setProgress(int progress) {this.progress = progress;}

    public void setFileSize(long fileSize) {this.fileSize = fileSize;}

    public void setDownloadedSize(long downloadedSize) {this.downloadedSize = downloadedSize;}
}

【讨论】:

  • 这应该是公认的答案,因为它完美地处理了进度条的变化。
【解决方案4】:

查看this thread which has a problem similar to yours。如果 wget 工作正常,那么服务器显然支持恢复下载。看起来您没有像上面链接的接受答案中提到的那样设置 If-Range 标头。 IE。添加:

// Initial download.
String lastModified = connection.getHeaderField("Last-Modified");

// ...

// Resume download.
connection.setRequestProperty("If-Range", lastModified); 

【讨论】:

  • 使用您的代码,我的应用不会下载任何内容。 lastmodified 字符串包含有效日期(2008 年 5 月 14 日)。会是什么?
【解决方案5】:

这个怎么样?

public static void download(DownloadObject object) throws IOException{
    String downloadUrl = object.getDownloadUrl();
    String downloadPath = object.getDownloadPath();
    long downloadedLength = 0;

    File file = new File(downloadPath);
    URL url = new URL(downloadUrl);

    BufferedInputStream inputStream = null;
    BufferedOutputStream outputStream = null;

    URLConnection connection = url.openConnection();

    if(file.exists()){
        downloadedLength = file.length();
        connection.setRequestProperty("Range", "bytes=" + downloadedLength + "-");
        outputStream = new BufferedOutputStream(new FileOutputStream(file, true));

    }else{
        outputStream = new BufferedOutputStream(new FileOutputStream(file));

    }

    connection.connect();

    inputStream = new BufferedInputStream(connection.getInputStream());


    byte[] buffer = new byte[1024*8];
    int byteCount;

    while ((byteCount = inputStream.read(buffer)) != -1){
        outputStream.write(buffer, 0, byteCount);
        break;

    }

    inputStream.close();
    outputStream.flush();
    outputStream.close();

}

使用break; 测试代码.. ;)

【讨论】:

  • 除了提供一些代码之外,请添加一些文字来解释为什么您的代码有效。
  • @buczek 是下载文件的标准代码,除了 connect.setRequestProperty 方法和 FileOutputStream 的 append 属性设置为 true。
  • 在邮递员中检查请求,您可能需要添加检查服务器是否部分或全部响应。我必须添加:connection.setRequestProperty("Accept-Encoding", "")
【解决方案6】:

我有办法让你的代码工作。

  1. 首先,检查文件是否存在
  2. 如果文件存在,设置连接:

    connection.setRequestProperty("Range", "bytes=" + bytedownloaded + "-");
    
  3. 如果文件不存在,请在新文件中进行相同的下载。

【讨论】:

    【解决方案7】:

    由于问题被标记为 Android: 您是否尝试过使用DownloadManager。 它可以很好地为您处理所有这些事情。

    【讨论】:

    • 谢谢,我改用它,现在下载应该稳定多了!
    • 记住它需要 API 级别 9 (Android 2.3)。我不知道这是否有问题。无论如何,2.2 即将推出。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2013-04-22
    • 2014-05-18
    • 1970-01-01
    相关资源
    最近更新 更多