23

如果服务没有完成,我会创建 1 分钟的延迟计时器来关闭服务。看起来像这样:

private Handler timeoutHandler = new Handler();

在 onCreate() 里面

timeoutHandler.postDelayed(new Runnable()
        {
            public void run()
            {
                Log.d(LOG_TAG, "timeoutHandler:run");

                DBLog.InsertMessage(getApplicationContext(), "Unable to get fix in 1 minute");
                finalizeService();
            }
        }, 60 * 1000);

如果我在这 1 分钟之前完成工作 - 我想取消这个延迟的事情,但不知道如何取消。

4

4 回答 4

47

你不能用匿名的 Runnable 真正做到这一点。将 Runnable 保存到命名变量怎么样?

Runnable finalizer = new Runnable()
    {
        public void run()
        {
            Log.d(LOG_TAG, "timeoutHandler:run");

            DBLog.InsertMessage(getApplicationContext(), "Unable to get fix in 1 minute");
            finalizeService();
        }
    };
timeoutHandler.postDelayed(finalizer, 60 * 1000);

...

// Cancel the runnable
timeoutHandler.removeCallbacks(finalizer);
于 2011-05-04T15:13:08.520 回答
15

如果您不想保留可运行对象的引用,您可以简单地调用:

timeoutHandler.removeCallbacksAndMessages(null);

官方文档说:

... 如果 token 为空,所有回调和消息都将被删除。

于 2014-12-24T16:47:10.970 回答
10

您可能想用这样的使用替换postDelayed使用sendMessageDelayed

private Handler timeoutHandler = new Handler(){
    @Override
    public void handleMessage(Message msg)
    {
            switch (msg.what){
        case 1:
            ((Runnable)msg.obj).run();
            break;
        }
    }
};

然后发消息:

Message m = Message.obtain();
m.what = 1;
m.obj = new Runnable(){
            public void run()
            {
                Log.d(LOG_TAG, "timeoutHandler:run");

                DBLog.InsertMessage(getApplicationContext(), "Unable to get fix in 1 minute");
                finalizeService();
            }
        };
timeoutHandler.sendMessageDelayed(m, 60 * 1000);

然后取消:

timeoutHandler.removeMessages(1);

无需跟踪可运行文件。

于 2011-05-04T15:13:58.810 回答
2

如果我在这 1 分钟之前完成工作 - 我想取消这个延迟的事情,但不知道如何取消。

使用Handler.removeCallbacks(yourRunnable).

于 2011-05-04T15:10:21.553 回答