自己造轮子,写个下载管理器(三)

DownloadManager.class

在完成了编写网络请求、下载进度、线程等关键“零件”过后,我们现在要做的就是把这些零件通过一个核心骨架串联,方便调用。这有点类似Java代理模式一样,需要一个或多个Class带负责这些组件的逻辑运行。现在,我们要编写一个完成这些职责的Class,这里就命名为DownloadManager.java

 /**
 * Created by mid1529 on 2016/12/19.
 * 下载管理方法
 */
public final class DownloadManager {
    private static final String TAG = "DownloadManager";
    public static final int HANDLER_DOWNLOAD_COMPLETED = 0x234123;
    public static final int HANDLER_DOWNLOAD_FAILED = 0x12345234;
    public static final int HANDLER_DOWNLOAD_PROGRESS = 0x2052344;
    public static final int HANDLER_DOWNLOAD_CALL = 0x4234134;
    @SuppressLint("StaticFieldLeak")
    private static DownloadManager ourInstance = new DownloadManager();
    private SQLiteDao mSQLiteDao = null;
    private ArrayMap<Long, DownloadTask> mDownloadTaskQueue = null; //当前下载任务的集合
    private ArrayMap<Long, Call> mDownloadTaskCallQueue = null;
    private DownloadService.DownloadServiceBinder mDownloadServiceBinder = null;
    private ServiceConnection mServiceConnection = null;
    private Application mApplication = null;
    private DownloadCallback mDownloadCallback = null;
    private DownloadHandler mHandler = null;
    private boolean mIsDeleteHistory = false;
    private ExecutorService mCachedThreadPool = null;


    public synchronized static DownloadManager getInstance() {
        if (ourInstance == null) {
            synchronized (DownloadManager.class) {
                ourInstance = new DownloadManager();
            }
        }
        return ourInstance;
    }

    private DownloadManager() {
        mDownloadTaskQueue = new ArrayMap<>();
        mDownloadTaskCallQueue = new ArrayMap<>();
    }

    /**
     * 在Application中调用此方法初始化
     * 只有调用这方法后才可以正常运行,否则会报错
     *
     * @param app Application
     * @return DownloadManager实例
     */
    public synchronized DownloadManager startService(Application app) {
        if (mServiceConnection != null && mApplication != null)
            return ourInstance;
        mHandler = new DownloadHandler();
        mCachedThreadPool = Executors.newCachedThreadPool();
        mApplication = app;
        Intent intent = new Intent(app, DownloadService.class);
        mServiceConnection = new ServiceConnection() {
            @Override
            public void onServiceConnected(ComponentName componentName, IBinder iBinder) {
                try {
                    mDownloadServiceBinder = (DownloadService.DownloadServiceBinder) iBinder;
                    mDownloadServiceBinder.setHandler(mHandler);
                } catch (Exception e) {
                    e.printStackTrace();
                    onDestory();
                }
            }

            @Override
            public void onServiceDisconnected(ComponentName componentName) {
                onDestory();
            }
        };
        app.startService(intent);
        app.bindService(intent, mServiceConnection, Context.BIND_AUTO_CREATE);
        mSQLiteDao = new SQLiteDao(app);
        return ourInstance;
    }

    /**
     * 添加到下载队列,阻塞线程,建议到子线程操作
     * 如果任务已存在队列或者任务的url、filename不合法,也会添加错误
     *
     * @param task 下载任务
     * @return 是否添加成功
     */
    public boolean addTaskToQueue(DownloadTask task) {
        if (!isDownloadTaskLegal(task)) //检测下载任务的参数是否合法
            return false;
        checkInited();
        DownloadTask query = mSQLiteDao.queryTask(task);
        task.setPause(false);
        if (query == null) {//不存在表中,执行插入操作,加入下载列表
            mSQLiteDao.insertDownloadTask(task);
            query = mSQLiteDao.queryTask(task);
            task.setTaskId(query.getTaskId());
            mDownloadTaskQueue.put(query.getTaskId(), query);
            mDownloadServiceBinder.startDownload(task);
        } else {
            task.setTaskId(query.getTaskId());
            if (mDownloadTaskQueue.get(task.getTaskId()) == null) {
                task.setPause(false);
                mDownloadTaskQueue.put(task.getTaskId(), task);
                mDownloadServiceBinder.startDownload(task);
            }
        }
        return true;
    }

    /**
     * 验证下载任务是否合法
     *
     * @param downloadTask 下载任务
     * @return 是否合法
     */
    @SuppressWarnings("RedundantIfStatement")
    private boolean isDownloadTaskLegal(DownloadTask downloadTask) {
        if (TextUtils.isEmpty(downloadTask.getFileName()))
            return false;
        if (TextUtils.isEmpty(downloadTask.getUrl()))
            return false;
        if (HttpUrl.parse(downloadTask.getUrl()) == null)
            return false;
        return true;
    }

    /**
     * 暂停下载,并将下载任务从容器中移除
     *
     * @param task 下载的任务
     */
    public void pauseDownload(DownloadTask task) {
        task.setPause(true);
        removeTaskFromQueue(task, false);
    }


    /**
     * 从下载列表中移除下载任务
     *
     * @param downloadTask      下载任务
     * @param isDeleteDbHistory 是否删除对应任务在数据库中的记录
     */
    private void removeTaskFromQueue(final DownloadTask downloadTask, boolean isDeleteDbHistory) {
        if (downloadTask.getTaskId() == null)
            return;
        DownloadTask task = mDownloadTaskQueue.get(downloadTask.getTaskId());
        if (task != null) {
            mDownloadTaskQueue.remove(task.getTaskId());
        }
        Call call = mDownloadTaskCallQueue.get(downloadTask.getTaskId());
        if (call != null) {
            call.cancel();
        }
        if (isDeleteDbHistory) { //不需要删除
            //noinspection ConstantConditions
            mCachedThreadPool.execute(new Runnable() {
                @Override
                public void run() {
                    mSQLiteDao.deleteDownloadTaskByKey(downloadTask.getTaskId());
                }
            });
        }
    }

    private void addCallToQueue(Long taskId, Call call) {
        mDownloadTaskCallQueue.put(taskId, call);
    }

    /**
     * 停止下载服务,将会停止所有,且清空下载队列,但不会删除数据库中下载记录
     */
    public void stopService() {
        checkInited();
        mDownloadTaskQueue.clear();
        removeDownloadCallback();
        if (mApplication == null)
            return;
        mApplication.stopService(new Intent(mApplication, DownloadService.class));
        mApplication.unbindService(mServiceConnection);
    }

    /**
     * 销毁
     * 清空Handler中的消息
     * 清空用到的容器
     * 将对象置为null
     * 尽量减少内存泄漏的可能
     */
    private void onDestory() {
        mHandler.removeCallbacksAndMessages(null);
        mSQLiteDao = null;
        mDownloadTaskQueue.clear(); //当前下载任务的集合
        mDownloadTaskCallQueue.clear();
        mDownloadServiceBinder = null;
        mServiceConnection = null;
        mCachedThreadPool.shutdown();
        mCachedThreadPool = null;
        mApplication = null;
    }

    /**
     * 检查是否已经调用start()方法初始化,没有则直接抛异常
     */
    private void checkInited() {
        if (mDownloadServiceBinder == null) {
            throw new NullPointerException("Do you remember invok \"DownloadManager.getInstance().start()\" in your Application.class? ");
        }
    }

    /**
     * @return 返回Dao
     */
    public List<DownloadTask> getDownloadTask() {
        checkInited();
        return mSQLiteDao.queryAllTask();
    }

    /**
     * 设置下载回调监听
     *
     * @param callback 回调
     */
    public void setDownloadCallback(DownloadCallback callback) {
        this.mDownloadCallback = callback;
    }

    /**
     * 移除监听
     */
    public void removeDownloadCallback() {
        this.mDownloadCallback = null;
    }

    /**
     * 更新下载任务到数据库
     *
     * @param downloadTask 下载任务
     */
    private void updateDownloadTask(final DownloadTask downloadTask) {
        mCachedThreadPool.execute(new Runnable() {
            @Override
            public void run() {
                mSQLiteDao.updateDownloadTask(downloadTask);
            }
        });
    }

    /**
     * 清空下载的信息
     *
     * @param downloadTask 下载任务
     */
    private void cleanDownloadInfo(final DownloadTask downloadTask) {
        mCachedThreadPool.execute(new Runnable() {
            @Override
            public void run() {
                removeTaskFromQueue(downloadTask, mIsDeleteHistory);
            }
        });

    }

    /**
     * 是否在下载失败或完成后,删除下载任务
     *
     * @return 是否删除
     */
    public boolean isDeleteHistory() {
        return mIsDeleteHistory;
    }

    /**
     * 设置是否删除下载任务的记录
     *
     * @param deleteHistory 是否删除
     */
    public void setDeleteHistory(boolean deleteHistory) {
        mIsDeleteHistory = deleteHistory;
    }

    /**
     * Handler负责处理进度传递消息等
     */
    @SuppressWarnings("unchecked")
    private static class DownloadHandler extends Handler {
        @Override
        public void handleMessage(Message msg) {
            switch (msg.what) {
                case HANDLER_DOWNLOAD_PROGRESS:
                    getInstance().updateDownloadTask((DownloadTask) msg.obj);
                    getInstance().mDownloadCallback.onProgress((DownloadTask) msg.obj);
                    break;
                case HANDLER_DOWNLOAD_COMPLETED:
                    getInstance().cleanDownloadInfo((DownloadTask) msg.obj);
                    getInstance().mDownloadCallback.onComplete((DownloadTask) msg.obj);
                    break;
                case HANDLER_DOWNLOAD_FAILED:
                    getInstance().cleanDownloadInfo((DownloadTask) msg.obj);
                    getInstance().mDownloadCallback.onDownloadFaild((DownloadTask) msg.obj);
                    break;
                case HANDLER_DOWNLOAD_CALL:
                    if (msg.obj instanceof ArrayMap) {
                        ArrayMap<Long, Call> map = (ArrayMap<Long, Call>) msg.obj;
                        if (map.size() != 0) {
                            getInstance().addCallToQueue(map.keyAt(0), map.get(map.keyAt(0)));
                        }
                    }
                    break;
            }
        }
    }

}

典型的,因为一个App中只需要一个下载管理实例,所以我们控制只能用单例模式获取DownloadManager实例,然后进行通信。有兴趣的参考注释阅读。

总结

这个项目中,在提取网络Header中的数据时,采用正则表达式提取,比如Range,这种做法总感觉不科学,但不知道有什么方法获取到,这里就谦虚谦虚,谁能告知下科学方法。
整体思路就是这样,现在只是V0.2版本,很多地方不是很严谨,很多地方偷懒,过两天再改善吧。

流程图及整体Project周日会放到GitHub中,有兴趣的可以看看

到这里算是结束了这为期三天的水文,能静下来写点东西还真不容易,好好坚持吧。

Lolipop
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,560评论 4 361
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,104评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,297评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,869评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,275评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,563评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,833评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,543评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,245评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,512评论 2 244
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,011评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,359评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,006评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,062评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,825评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,590评论 2 273
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,501评论 2 268

推荐阅读更多精彩内容