Android:创建一个后台线程,定期运行并执行UI任务



OK,所以我知道如何做后台任务,我知道如何做周期性任务(使用处理postdelayed和runnable),我也知道如何从后台线程做UI任务(通过处理程序),但我不能执行周期性后台任务,在UI线程上做一些动作。

我正试图每分钟执行一些后台任务,其中我必须打一个网络电话。调用结束后,根据输出,我必须更新UI。我试过这样做

private void DoTask() {
        Thread thread = new Thread() {
            public void run() {
                Looper.prepare();
                final Handler handler = new Handler();
                handler.post(netRunnable);
                Looper.loop();
            }
        };
        thread.start();
}
Runnable netRunnable = new Runnable() {
    @Override
    public void run() {
        handler.getLooper().prepare();
        final Handler handler1 = new Handler(Looper.getMainLooper());
        if ( do background work and check result){
            handler1.post(new Runnable() {
                @Override
                public void run() {
                    //Do UI Task
                }
            });
        }
        handler.getLooper().loop();
        handler.postDelayed(netRunnable, 60000);
    }
}

我知道我的实现可能有一些根本性的缺陷,但我不知道如何正确地完成这项任务。现在它给出的错误是Only one Looper may be created per thread,我明白它想说什么了。但是谁能告诉我正确的方法呢?

你可以使用异步任务。这些都是为它设计的:

http://developer.android.com/reference/android/os/AsyncTask.html

它允许你在后台执行一个网络调用,然后当你得到结果时,在UI线程上执行一个动作

声明:

 private class MyTask extends AsyncTask<Input, Void, Output> {
 protected Output doInBackground(Input... inputs) {
       // do something on the network
       return myOutput;// use this to transmit your result
 }
 protected void onPostExecute(Output result) {
     // do something on UI thread with the result
 }

}

如果你想重复它,只需创建一个可运行的启动它,并在每次调用后,安排下一个:

MyTask myTask;
Handler handler = new Handler();
    Runnable myRunnable = new Runnable() {
            @Override
            public void run() {
                MyTask myTask = new MyTask();
                myTask.execute(myArg);
                handler.postDelayed(netRunnable, 60000); // schedule next call
            }
        }

第一次启动:

handler.postDelayed(myRunnable, 60000);

或者,如果您想立即启动它:

handler.post(myRunnable);

当activity被销毁时,不要忘记取消Task:

myTask.cancel(true);

也许你最好创建一个单独的(Intent)Service,并定期使用postDelayed调用它。在Activity中创建BroadcastReceiver,并在那里处理UI更改。

处理来自其他线程的UI更改的另一个提示:这是不可能的。因此你需要调用runonuthread。下面是如何使用它

如果活动频繁切换,为什么不反转责任呢?您可以创建一个执行周期性网络任务的服务。

,-您的活动定期调用此服务以获取值。-或者你使用一个监听器系统:你创建一个接口,你的活动必须实现,以便从任务完成

获得通知

最新更新