封装异步任务

项目中有一些操作比较耗时,例如AES解密,在第一次解密初始化时会用时1~2秒,这时候如果不把耗时操作放到异步任务里面,app就会出现屏幕假死、卡顿。放到异步任务一般人的做法是在每一个需要创建异步任务的地方创建一个类继承AsyncTask,把耗时操作放到doInBackground方法里面,处理完毕则在onPostExecute方法更新UI。这样的做法太繁琐,且创建出来的AsyncTask线程池不好管理,所以就考虑将异步任务封装出来,队列化管理。
首先创建ExecuteTaskManager,它是管理线程池,实际执行异步任务操作的类:

public class ExecuteTaskManager implements Runnable {

    /**
     * 线程执行完事儿后默认的回调类型
     */
    private static final int COMMON_EXCUTE_TASK_TYPE = 0;
    /**
     * 线程开关
     */
    public volatile boolean isRunning = false;
    /**
     * 是否初始化完成的开关
     */
    private boolean isHasInit = false;
    /**
     * 默认线程池的线程数量
     */
    private static final int DEFAULT_THREAD_NUM = 5;
    /**
     * 初始化时的线程数量
     */
    private int threadNum = DEFAULT_THREAD_NUM;
    /**
     * 定义一个单线程的线程池,专门用来执行耗时且不需要回调的操作
     */
    private static ScheduledExecutorService singlePool = null/*Executors.newSingleThreadScheduledExecutor()*/;
    /**
     * 定义一个大小为5的线程池(这个我们比较适合多个图片下载时使用)
     */
    private static ExecutorService threadPool = null/*Executors.newFixedThreadPool(threadNum)*/;
    /**
     * 任务执行队列
     */
    private static ConcurrentLinkedQueue<ExecuteTask> allExecuteTask = null/*new ConcurrentLinkedQueue<ExcuteTask>()*/;
    /**
     * 回调接口列表
     */
    private static ConcurrentHashMap<Integer, Object> uniqueListenerList = null/*new ConcurrentHashMap<String, Object>()*/;


    public Handler getHandler() {
        return handler;
    }

    public int getThreadNum() {
        return threadNum;
    }

    public boolean isHasInit() {
        return isHasInit;
    }

    public boolean isRunning() {
        return isRunning;
    }


    /**
     * @author Rocky
     * @desc 得到普通的 ExcuteTask 对象,
     * 对外界开放的回调接口
     */
    public interface GetExcuteTaskCallback {
        void onDataLoaded(ExecuteTask task);
    }


    /**
     * 直接把数据发送到主线程
     */
    private final static Handler handler = new Handler(Looper.getMainLooper()) {
        @Override
        public void handleMessage(Message msg) {
            switch (msg.what) {
                case COMMON_EXCUTE_TASK_TYPE:
                    if (msg.obj != null
                            && msg.obj instanceof ExecuteTask) {
                        ExecuteTaskManager.getInstance().doCommonHandler((ExecuteTask) msg.obj);
                    }
                    break;
            }
        }
    };


    private static ExecuteTaskManager instance = null;

    private ExecuteTaskManager() {
    }

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

    /**
     * 初始化操作,这个主要是初始化需要执行异步
     * 回调任务的线程池,默认开启5个线程
     */
    public void init() {
        init(threadNum);
    }

    /**
     * 初始化操作,这个主要是初始化需要执行异步
     * 回调任务的线程池,可以传入线程的个数
     */
    public synchronized void init(int initNum) {
        if (!isHasInit) {
            /**
             * 初始化之后就相当于开始了线程次的运行
             * 只不过如果没有任务处于等待状态
             */
            isRunning = true;
            if (initNum > 0) {
                threadNum = initNum;
            }
            threadPool = Executors.newFixedThreadPool(threadNum);
            singlePool = Executors.newSingleThreadScheduledExecutor();
            allExecuteTask = new ConcurrentLinkedQueue<>();
            uniqueListenerList = new ConcurrentHashMap<>();

            /**
             * 初始化需要用到的线程
             */
            for (int i = 0; i < threadNum; i++) {
                threadPool.execute(this);
            }
            isHasInit = true;
        }
    }


    /**
     * 当应用被销毁时,执行清理操作
     */
    public void doDestory() {
        /**
         * 关闭线程开关
         */
        isRunning = false;
        isHasInit = false;
        if (allExecuteTask != null) {
            allExecuteTask.clear();
            allExecuteTask = null;
        }
        if (uniqueListenerList != null) {
            uniqueListenerList.clear();
            uniqueListenerList = null;
        }
        if (threadPool != null) {
            threadPool.shutdown();
            threadPool = null;
        }
        if (singlePool != null) {
            singlePool.shutdown();
            singlePool = null;
        }
    }

    /**
     * 向任务队列中添加任务对象,添加成功后,
     * 任务会自动执行,执行完事儿后,不进行任何回调操作
     *
     * @param task 可执行的任务对象
     */
    public void newExcuteTask(ExecuteTask task) {
        if (task != null) {
            allExecuteTask.offer(task);
            synchronized (allExecuteTask) {
                allExecuteTask.notifyAll();
            }
        }
    }

    /**
     * 这个方法主要是获取普通的回调数据,
     * 获取成功后会把加入的 ExcuteTask 对象回调到用户界面
     *
     * @param task     加入的任务Task
     * @param callback 任务的回调接口GetDataCallback
     */
    public void getData(ExecuteTask task, GetExcuteTaskCallback callback) {
        /**
         *  把CallBack 接口加入列表中,用完之后移除
         */
        try {
            if (task != null && callback != null) {
                if (task.getUniqueID() == 0) {
                    task.setUniqueID(task.hashCode());
                }
                uniqueListenerList.put(task.getUniqueID(), callback);

                /**
                 * 开始加入任务,执行任务
                 */
                newExcuteTask(task);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    /**
     * 从任务队列中移除任务对象,使其不再执行(如果任务已经执行,则此方法无效)
     *
     * @param task 添加的任务对象
     */
    public void removeExcuteTask(ExecuteTask task) {
        if (task != null) {
            uniqueListenerList.remove(task.getUniqueID());
            allExecuteTask.remove(task);
        }
    }


    public void log() {
    }


    /**
     * 清除所有的任务
     */
    public void removeAllExcuteTask() {
        allExecuteTask.clear();
        uniqueListenerList.clear();
    }


    //=================================任务执行、回调、分发start============================================

    /**
     * 所有的异步任务都在此执行
     */
    @Override
    public void run() {
        // TODO Auto-generated method stub
        while (isRunning) {

            /**
             * 从allExcuteTask取任务
             */
            ExecuteTask lastExecuteTask = allExecuteTask.poll();

            if (lastExecuteTask != null) {
                try {
                    /**
                     * 真正开始执行任务,
                     * 所有的耗时任务都是在子线程中执行
                     */
                    doExcuteTask(lastExecuteTask);
                } catch (Exception e) {
                    // TODO: handle exception
                    e.printStackTrace();
                    /**
                     * 处理异常的回调
                     */
                    lastExecuteTask.setStatus(ExecuteTask.EXCUTE_TASK_ERROR);
                    if (lastExecuteTask.isMainThread()) {
                        Message msg = Message.obtain();
                        msg.what = COMMON_EXCUTE_TASK_TYPE;
                        msg.obj = lastExecuteTask;
                        handler.sendMessage(msg);
                    } else {
                        doCommonHandler(lastExecuteTask);
                    }
                    uniqueListenerList.remove(lastExecuteTask.getUniqueID());
                }
            } else {
                try {
                    synchronized (allExecuteTask) {
                        allExecuteTask.wait();
                    }
                } catch (InterruptedException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }
            }
        }
    }


    /**
     * 根据不同的ExcuteTask,执行相应的任务
     * <p>
     * 这个是真正开始执行异步任务的地方,
     * 即调用需要在子线程执行的代码==>task.doTask()
     *
     * @param task ExcuteTask对象
     */
    private void doExcuteTask(ExecuteTask task) {

        ExecuteTask result = task.doTask();

        /**
         *
         * 开始执行的Task和最后得到的Task是同一个的时候,才会进行回调,
         * 否则不进行回调(保证在回调得到数据的时候知道是哪一个Task,以便进行强转)
         *
         *
         * 没有UniqueID相当于不需要回调
         *
         */
        if (result != null && task == result && result.getUniqueID() != 0) {
            /**
             *  发送当前消息,更新UI(把数据回调到界面),
             *  下面不用做任何的发送消息,
             *  只在这一个地方发送就行,否者会发生错误!
             */

            if (result.isMainThread()) {
                Message msg = Message.obtain();
                msg.what = COMMON_EXCUTE_TASK_TYPE;
                msg.obj = result;
                handler.sendMessage(msg);
            } else {
                doCommonHandler(task);
            }
        } else {
            uniqueListenerList.remove(task.getUniqueID());
        }
    }

    /**
     * 真正的回调操作,所有的任务在这里
     * 把数据回调到主界面
     *
     * @param task ExcuteTask对象
     */
    private void doCommonHandler(ExecuteTask task) {
        if (task != null && uniqueListenerList.containsKey(task.getUniqueID())) {
            try {
                /**
                 * 回调整个Task数据
                 * 然后可以回调方法中去直接更新UI
                 */
                ((GetExcuteTaskCallback) uniqueListenerList.get(task.getUniqueID())).onDataLoaded(task);
                /**
                 * 回调完成移除CallBack对象
                 */
                uniqueListenerList.remove(task.getUniqueID());
            } catch (Exception e) {
                // TODO: handle exception
                e.printStackTrace();
            }
        }
    }
    /**=================================任务执行、回调、分发end============================================*/


    /**=================================单线程池,可以顺序,延迟执行一些任务start============================================*/

    /**
     * 顺序执行耗时的操作
     *
     * @param runnable 对象
     */
    public void excute(Runnable runnable) {
        singlePool.execute(runnable);
    }

    /**
     * 顺序执行耗时的操作
     *
     * @param runnable 对象
     * @param delay    延迟执行的时间,单位毫秒
     */
    public void excute(Runnable runnable, long delay) {
        singlePool.schedule(runnable, delay, TimeUnit.MILLISECONDS);
    }

    /**
     * 顺序执行耗时的操作
     *
     * @param runnable 对象
     * @param delay    延迟执行的时间
     * @param timeUnit 时间单位
     */
    public void excute(Runnable runnable, long delay, TimeUnit timeUnit) {
        singlePool.schedule(runnable, delay, timeUnit);
    }

    public void scheduleAtFixedRate(Runnable runnable, long delay, long period, TimeUnit timeUnit) {
        singlePool.scheduleAtFixedRate(runnable, delay, period, timeUnit);
    }

    public void scheduleAtFixedRate(Runnable runnable, long delay, long period) {
        singlePool.scheduleAtFixedRate(runnable, delay, period, TimeUnit.MILLISECONDS);
    }

    public void scheduleAtFixedRate(Runnable runnable, long period) {
        singlePool.scheduleAtFixedRate(runnable, 0, period, TimeUnit.MILLISECONDS);
    }
    /**=================================单线程池,可以顺序,延迟执行一些任务end============================================*/
}

还有异步任务类,负责存储单个任务的数据,任务处理的结果可以从该类中读取。

public abstract class ExecuteTask implements Runnable, Serializable {

    public static final int EXCUTE_TASK_ERROR = -1;

    /**
     * 这个会自动生成,不用自己设置
     */
    protected int uniqueID;
    /**
     * 主要是用来初始化的时候传入参数,
     * 然后根据不用的参数来进行异步操作
     */
    @SuppressWarnings("rawtypes")
    protected Map taskParam;// 内容参数
    /**
     * 异步操作完成之后的状态,失败、成功 or 其他
     */
    protected int status;
    /**
     * 如果是网络请求,并且获取的数据是Json,
     * 则直接可以给此字段赋值,然后在回调中get Json数据
     */
    protected String json;
    /**
     * 这个是异步操作后,如果想把异步数据传到UI线程,
     * 则可以通过此字段赋值,然后再强转得到所要的数据
     */
    protected Object result;

    private boolean isMainThread = Looper.myLooper() == Looper.getMainLooper();

    public ExecuteTask() {
    }

    public int getUniqueID() {
        return uniqueID;
    }

    public void setUniqueID(int uniqueID) {
        this.uniqueID = uniqueID;
    }

    @SuppressWarnings("rawtypes")
    public Map getTaskParam() {
        return taskParam;
    }

    @SuppressWarnings("rawtypes")
    public void setTaskParam(Map taskParam) {
        this.taskParam = taskParam;
    }

    public int getStatus() {
        return status;
    }

    public void setStatus(int status) {
        this.status = status;
    }

    public String getJson() {
        return json;
    }

    public void setJson(String json) {
        this.json = json;
    }

    public Object getResult() {
        return result;
    }

    public void setResult(Object result) {
        this.result = result;
    }


    public boolean isMainThread() {
        return isMainThread;
    }

    public void setIsMainThread(boolean isMainThread) {
        this.isMainThread = isMainThread;
    }

    @Override
    public void run() {
        doTask();
    }

    /**
     * 专门用来执行耗时的操作,
     * 子类只需要继承此类,实现此方法,
     * 在这个方法中执行所有耗时的操作
     * 用ExcuteTaskManager进行执行,
     * 可以回调, 也可以不回调
     *
     * 在继承此类的时候 doTask
     * return null(不再回调) or  return this(会回调)
     *
     * @return
     */
    public abstract ExecuteTask doTask();
}

然后将自己的业务逻辑放到自定义的异步任务中,

public class AESDecryptTask extends ExecuteTask {

    private String forDecryptStr;

    public String getForDecryptStr() {
        return forDecryptStr;
    }

    public void setForDecryptStr(String forDecryptStr) {
        this.forDecryptStr = forDecryptStr;
    }

    @Override
    public ExecuteTask doTask() {
        try {
            String str = AESUtil.decrypt(forDecryptStr); //这是比较耗时的AES解密方法,主要是初始化的步骤比较耗时
            setResult(str);
        } catch (UnsupportedEncodingException e) {
            e.printStackTrace();
        }

        return this;
    }
}

为了方便使用,把创建异步任务的逻辑放到工具类里面,

public static AESDecryptTask decryptAsync(String forDecryptStr, ExecuteTaskManager.GetExcuteTaskCallback callback){
        AESDecryptTask task = new AESDecryptTask();
        task.setForDecryptStr(forDecryptStr);
        ExecuteTaskManager.getInstance().getData(task, callback);

        return task;
}

需要调用的时候直接调用工具类方法,它会返回异步任务示例本身,方便做一些销毁操作,在callback里面写更新UI的逻辑:

decryptTask = AESUtil.decryptAsync(user.phone, new ExecuteTaskManager.GetExcuteTaskCallback() {
                @Override
                public void onDataLoaded(ExecuteTask task) {
                    try {
                        String result = (String) task.getResult();
                        if (TextUtils.isEmpty(result)){
                            result = "";
                        }

                        selectedUserPhone = result;
                        String userName = StringUtil.hideSeveralNumber(selectedUserPhone);
                        et_userName.setText(userName);
                        et_userName.setSelection(userName.length());

                        et_userName.addTextChangedListener(new TextWatcher() {
                            @Override
                            public void beforeTextChanged(CharSequence s, int start, int count, int after) {
                            }

                            @Override
                            public void onTextChanged(CharSequence s, int start, int before, int count) {
                            }

                            @Override
                            public void afterTextChanged(Editable s) {
                                if (isAutoComplete && !isSelectedFromHint) {
                                    //如果当前用户名为从下拉列表选择,用户对用户名进行编辑,则清空输入框
                                    s.clear();
                                    iv_clearUserName.setVisibility(View.INVISIBLE);
                                    iv_userPhoto.setImageResource(R.mipmap.avatar);
                                    selectedUserPhone = "";
                                    isAutoComplete = false;
                                } else {
                                    //如果当前用户名为正常输入,则进行是否显示清空按钮的处理
                                    String text = s.toString();
                                    if (!TextUtils.isEmpty(text)) {
                                        iv_clearUserName.setVisibility(View.VISIBLE);
                                    } else {
                                        iv_clearUserName.setVisibility(View.INVISIBLE);
                                    }
                                }

                                isSelectedFromHint = false;
                                //每当输入内容发生变化,都调用此方法,判断登录按钮是否可点击
                                checkLoginButtonEnable();
                            }
                        });
                    } catch (Exception e) {
                        e.printStackTrace();
                    } finally {
                        enableUsernameEditText();
                    }

                }
            });

为了避免退出Activity后异步任务还在执行,而重新进入Activity异步任务又会重复执行,在onDetroy方法中销毁当前的异步任务:

@Override
    protected void onDestroy() {
        ......
        ExecuteTaskManager.getInstance().removeExcuteTask(decryptTask);
        super.onDestroy();
    }

这样写的好处是异步任务进一步抽象,代码比较简洁,由统一的线程池集中管理,在销毁时可以很容易地取消正在执行的异步任务。

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

推荐阅读更多精彩内容