Http分段下载实现

6 篇文章 0 订阅
2 篇文章 1 订阅

前言

最近要在安卓里面下载大文件,考虑到现场网络和使用的情况,如果简单http下载的话,很有可能下载崩溃或者中断失败。
原本想用wget做断点续传的,这样就简单很多,但是安卓不给运行wget


一、什么是分段下载?

简单来说就是把文件分拆开来,一块一块的下载,这样速度会有所损失,但胜在可控、健壮
在http里面,主要是请求头:Range

下面看一下Range字段常用表示的写法:
Range: bytes=0-1024 获取最前面1025个字节
Range: bytes=-500   获取最后500个字节
Range: bytes=1025-  获取从1025开始到文件末尾所有的字节
Range: 0-0          获取第一个字节
Range: -1           获取最后一个字节
例如,在一个请求头中有Range:byte=0-1024,那么表示的意思就是请求数据的前1025个字节。
 如果这个分段请求的返回码是206,并且指示的分段范围是0-1024,文件的总大小是7877,那么在响应头中的数据应该表示为:
Content-Range: bytes 0-1024/7877

二、使用方法

直接上代码

package com.kl;

import com.kl.utils.IOUtils;
import com.kl.utils.Logx;

import java.io.InputStream;
import java.io.RandomAccessFile;
import java.net.HttpURLConnection;
import java.net.URL;

class _DownLoadThread implements Runnable
{
    private String _url = "";
    private String _local = "";

    private long _len = 0;

    public _DownLoadThread(String url, String local, long len)
    {
        _url = url;
        _local = local;

        _len = len;

        new Thread(this).start();
    }

    @Override
    public void run()
    {
        long start, end, n = 0;
        long size = 1024 * 1024 * 50;

        start = 0;
        _len--;

        Logx.log("load: " + _url);
        while (start < _len)
        {
            Logx.log("load process: " + start);
            end = Math.min(start + size-1, _len);
            if( down(start, end) )
            {
                n = 0;
                start += size;
            }
            else n++;

            if( n > 10 )
            {
                Logx.log("error, retry too many: " + _url);
                break;
            }
        }
        if( n==0 )
        {
            Logx.log("load complete");

            NoticeUIChange.ins().newOk();
        }

        NoticeUIChange.ins().removeNew();
    }

    private boolean down(long startIndex, long endIndex)
    {
        HttpURLConnection conn = null;
        InputStream is = null;
        RandomAccessFile raf = null;
        try
        {
            URL url = new URL(_url);
            conn = (HttpURLConnection) url.openConnection();
            conn.setRequestMethod("GET");
            //请求服务器下载部分的文件的指定位置
            conn.setRequestProperty("Range", "bytes=" + startIndex + "-" + endIndex);
            conn.setConnectTimeout(5000);

            is = conn.getInputStream();//返回资源
            raf = new RandomAccessFile(_local, "rwd");

            raf.seek(startIndex);//定位文件

            int len = 0;
            byte[] buffer = new byte[1024];
            while ((len = is.read(buffer)) != -1) {
                raf.write(buffer, 0, len);
            }
            is.close();
            raf.close();

            is = null;
            raf = null;

            return true;
        }
        catch (Exception e)
        {
            e.printStackTrace();
        }
        finally
        {
            if( is != null ) IOUtils.closeQuietly(is);
            if(raf != null) IOUtils.closeQuietly(raf);

            if( conn != null )
            {
                try{ conn.disconnect(); } catch (Exception e){}
            }
        }

        return false;
    }
}

里面的Logx类是日志类,可以都删掉或者替换自己的
IOUtils:

public class IOUtils
{
    public static void closeQuietly(Closeable close)
    {
        if( close != null )
        {
            try
            {
                close.close();
            }
            catch (Exception er)
            {
            }
        }
    }
}

总结

本文实现的是单线程下载,考虑现场的情况,多线程反而会出问题。如果想要多现场下载,改装下下载类就行

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值