(1)我们需要如何去下载文件,因为很多时候因为网络差、程序被kill掉等原因,文件无法一次下载完,我们需要继续发起下载文件的HTTP请求,这个时候不需要从头开始下,只需要告诉服务器,我们要从哪里开始下,哪里结束,可以通过设置“Range”字段的属性来实现。例如:

httpUrl.setRequestProperty("Range", "bytes=" + mCurrentDownloadLength + "-" + mFileTotalSize);



       (2)需要保存文件的下载进度,这个我最开始尝试着使用SharedPreferences保存进度,经过测试后发现使用该方法保存的进度不一定准确,可能的原因是在保持的时候程序中断了,比较可靠的方式是获取文件的大小,用.length()方法。

        (3)如何从指定的位置开始写文件,使用RandomAccessFile即可。

       (4)由于在弱网环境下,下载的文件可能会出错,所以对下载的文件需要进行MD5值校验。

       (5)此外,我们下载时,也需要知道文件的总大小,这样不用每次启动下载时,都进行MD5值校验是否下载完成,这个可以通过HTTP的请求实现。

具体的实现代码不复杂,部分代码如下所示:

public class DownloadTask extends Thread {
    
    private URL mDownloadUrl;
    private Activity mActivity;
    
    private String mZipFilePathStr;
    // 保存用户设置的so目录
    public static String mUserBaiduNaviSDK_SO = null;

    // 需要下载的文件的总大小
    private int mFileTotalSize;
    private int mCurrentDownloadLength = 0;
    
    private DownloadCallBack mCallback;
    // 每次读取的文件大小
    private int mBlockSize = 4 * 1024;
    
    private SharedPreferences mSharedPref;
    
    private int timeout = 50 * 1000; // 连接超时时间
    
    public DownloadTask(Activity activity, String sdcardRootPath, String appFolderName) {
        this.mActivity = activity;
    }
    
    public DownloadTask(Activity activity, String sdcardRootPath, String appFolderName, URL downloadUrl,
            DownloadCallBack callback) {
        this.mActivity = activity;
        this.mDownloadUrl = downloadUrl;
        this.mCallback = callback;
    }
    
    @Override
    public void run() {
        mSharedPref = mActivity.getPreferences(Context.MODE_PRIVATE);
        // String str = sdcardRootPath + File.separator + appFolderName + File.separator + "1.txt";
        init();
        
        if (mCurrentDownloadLength < mFileTotalSize) {
            try {
                HttpURLConnection httpUrl = (HttpURLConnection) mDownloadUrl.openConnection();
                if (httpUrl == null) {
                    return ;
                }
                setRequest(httpUrl);
                downloadFile(httpUrl);
                mCallback.downloadWorkOver();
            } catch (Exception e) {
                return ;
            }
        }
    }
    
    private void init() {
        mFileTotalSize = getTotalFileSize();
        mCurrentDownloadLength = getCurrentDownloadLength();
    }
    
    /*
     * 
     */
    private int getCurrentDownloadLength() {
        File file = new File(ZipUtils.getTmpZipFile());
        mCurrentDownloadLength = file.length();
        LogUtil.e("dingbbin", "dingbbin getCurrentDownloadLength is " + mCurrentDownloadLength);
        return mCurrentDownloadLength;
    }
    
    /*
     * 获取需要更新的文件的大小
     */
    private int getTotalFileSize() {
    //    return 8038626;
        HttpURLConnection urlConnection = null;
        int length = 0;
        try {
            urlConnection = (HttpURLConnection) mDownloadUrl.openConnection();
            urlConnection.setRequestProperty("Accept", "image/gif, image/jpeg, image/pjpeg, image/pjpeg,"
                    + " application/x-shockwave-flash, application/xaml+xml, application/vnd.ms-xpsdocument,"
                            + " application/x-ms-xbap, application/x-ms-application, application/vnd.ms-excel,"
                                + " application/vnd.ms-powerpoint, application/msword, */*");
            urlConnection.setRequestProperty("Referer", mDownloadUrl.toString());
            urlConnection.setRequestProperty("Connection", "Keep-Alive");
            urlConnection.setRequestProperty("Accept-Encoding", "identity");
            int status = urlConnection.getResponseCode();
            if (status == 200) {
                length = urlConnection.getContentLength();
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (urlConnection != null) {
                urlConnection.disconnect();
            }
        }
       
        return length;
    }
    
    private void setRequest(HttpURLConnection httpUrl) throws ProtocolException {
        httpUrl.setConnectTimeout(timeout);
        httpUrl.setRequestMethod("GET"); // 设置请求方法
        httpUrl.setRequestProperty("Accept", "image/gif, image/jpeg, image/pjpeg, image/pjpeg,"
                + " application/x-shockwave-flash, application/xaml+xml, application/vnd.ms-xpsdocument,"
                        + " application/x-ms-xbap, application/x-ms-application, application/vnd.ms-excel,"
                            + " application/vnd.ms-powerpoint, application/msword, */*");
        httpUrl.setRequestProperty("Referer", mDownloadUrl.toString());
        httpUrl.setRequestProperty("Charset", "UTF-8");
        httpUrl.setRequestProperty("Accept-Encoding", "identity");
        
        // 设置开始和结束范围
        // int endPos = fileTotalSize;
        httpUrl.setRequestProperty("Range", "bytes=" + mCurrentDownloadLength + "-" + mFileTotalSize);
        
        httpUrl.setRequestProperty("Connection", "Keep-Alive");
    }
    
    private void downloadFile(HttpURLConnection httpUrl) {
        InputStream inStream = null;
        RandomAccessFile accessFile = null;
        try {
            inStream = httpUrl.getInputStream();
            byte[] buffer = new byte[mBlockSize];
            int offset = 0;
            accessFile = new RandomAccessFile(ZipUtils.getTmpZipFile(), "rwd");
            // 定位到pos位置
            accessFile.seek(mCurrentDownloadLength);
            while ((offset = inStream.read(buffer, 0, mBlockSize)) != -1) {
                accessFile.write(buffer, 0, offset);
                // 更新已经下载的大小
                mCurrentDownloadLength += offset;
                LogUtil.e("dingbbin", "dingbbin current length is " + mCurrentDownloadLength);
            }
            if (mCurrentDownloadLength >= mFileTotalSize) {
                // download finish, rename file
                ZipUtils.getTmpZipFile().renameTo(ZipUtils.getZipFile());
                resetLength();
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (null != inStream) {
                try {
                    inStream.close();
                } catch (IOException e) {
                    LogUtil.e("", e.toString());
                }
            }
            if (null != accessFile) {
                try {
                    accessFile.close();
                } catch (IOException e) {
                    LogUtil.e("", e.toString());
                }
            }
            if (httpUrl != null) {
                httpUrl.disconnect();
            }
        }
    }
    
    private void resetLength() {
        Editor editor = mSharedPref.edit();
        editor.putInt(DynamicLoadConstants.NAVI_DOWNLOAD_LENGTH, 0);
        editor.commit();
    }
    
    private void unzipFile() {
        try {
            ZipUtils.decompress(ZipUtils.getZipFile(), new File(ZipUtils.getDownloadSoDir()));
        } catch (Exception e) {
            e.printStackTrace();
        }
        
        // 删除下载的文件
        ZipUtils.getZipFile().delete();
    }
    
    public interface DownloadCallBack {
        public void downloadWorkOver();
    }
}


需要注意的地方有2个,就是我们在获取文件总大小和下载文件时,发下Java和Android的实现不太一样,具体来说,同样的代码,同样的API,在Java环境中可以正常获取大小,可以实现断点续传,但是在Android上就不行,需要设置urlConnection.setRequestProperty("Accept-Encoding", "identity")才可以,具体的原因我也是不甚明白。