android - 如何处理 Timer was cancelled 错误?

标签 android

我用的是安卓。 最近我编码重新发送数据。 但发生 illegalstateexception 定时器被取消。

这是我的日志:

java.lang.RuntimeException: Unable to start receiver kr.co.iosystem.blackeyeonandroid.util.NetworkChangeReceiver: java.lang.IllegalStateException: Timer was canceled
                                                     at android.app.ActivityThread.handleReceiver(ActivityThread.java:2414)
                                                     at android.app.ActivityThread.access$1700(ActivityThread.java:135)
                                                     at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1272)
                                                     at android.os.Handler.dispatchMessage(Handler.java:102)
                                                     at android.os.Looper.loop(Looper.java:136)
                                                     at android.app.ActivityThread.main(ActivityThread.java:5001)
                                                     at java.lang.reflect.Method.invokeNative(Native Method)
                                                     at java.lang.reflect.Method.invoke(Method.java:515)
                                                     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:785)
                                                     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:601)
                                                     at dalvik.system.NativeStart.main(Native Method)
                                                  Caused by: java.lang.IllegalStateException: Timer was canceled
                                                     at java.util.Timer.scheduleImpl(Timer.java:561)
                                                     at java.util.Timer.schedule(Timer.java:459)
                                                     at kr.co.iosystem.blackeyeonandroid.sender.DataSender.sendPicData(DataSender.java:251)
                                                     at kr.co.iosystem.blackeyeonandroid.sender.DataSender.restart(DataSender.java:425)
                                                     at kr.co.iosystem.blackeyeonandroid.sender.DataSender.update(DataSender.java:337)
                                                     at java.util.Observable.notifyObservers(Observable.java:138)
                                                     at kr.co.iosystem.blackeyeonandroid.util.AutoObservable.notifyObservers(AutoObservable.java:13)
                                                     at kr.co.iosystem.blackeyeonandroid.main.Agency.notifyMessage(Agency.java:88)
                                                     at kr.co.iosystem.blackeyeonandroid.util.NetworkChangeReceiver.onReceive(NetworkChangeReceiver.java:200)
                                                     at android.app.ActivityThread.handleReceiver(ActivityThread.java:2407)
                                                     at android.app.ActivityThread.access$1700(ActivityThread.java:135) 
                                                     at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1272) 
                                                     at android.os.Handler.dispatchMessage(Handler.java:102) 
                                                     at android.os.Looper.loop(Looper.java:136) 
                                                     at android.app.ActivityThread.main(ActivityThread.java:5001) 
                                                     at java.lang.reflect.Method.invokeNative(Native Method) 
                                                     at java.lang.reflect.Method.invoke(Method.java:515) 

我认为这是发生异常的部分

private Timer mTimer = null; //global variable

@Override 
public void run() {
mTimer = new Timer();
...
}
@Override
public void update(){
if (this.senderState == SenderState.STOPPED) {
            this.mTimer.cancel();
    this.restart();   //Timer exception ...
        }

我能知道我哪里出错了吗?

最佳答案

我个人不喜欢使用 java.util.Timer 类。这主要是因为当你cancel()它时它不能重新启动。

通常我会使用 javax.swing.Timer 但我认为您不能在 android 中使用它。在 Android 中,我认为您应该创建自己的定时器类来封装 android.os.Handler 类。

这就是你的做法:

Handler 类包含一个名为postDelayed 的方法。它将延迟运行 Runnable

假设您想创建一个每 1000 毫秒运行一次的计时器,您可以创建一个处理程序并调用 postDelayed(someRunnable, 1000)

然后有趣的部分来了。上面的 someRunnable 应该在执行任何你喜欢的事情之后调用 postDelayed。这样就形成了一个“循环”。

“但是我怎样才能停止处理程序?我怎样才能在它启动后更改间隔?”你问了。

要向计时器类添加暂停方法,只需创建一个名为 paused 的字段并在调用 runnable 中的 postDelayed 之前检查其值。然后,您可以添加用于设置和获取 paused 值的 setter 和 getter。

要在计时器开始时更改间隔,只需创建另一个名为 interval 的字段并以类似的方式实现它。

长话短说

我很好,你知道的。这是要复制的代码:

import android.os.Handler;

public class Timer {
    private Handler handler;
    private boolean paused;

    private int interval;

    private Runnable task = new Runnable () {
        @Override
        public void run() {
            if (!paused) {
                Timer.this.runnable.run ();
                Timer.this.handler.postDelayed (this, interval);
            }
        }
    };

    private Runnable runnable;

    public int getInterval() {
        return interval;
    }

    public void setInterval(int interval) {
        this.interval = interval;
    }

    public void startTimer () {
        paused = false;
        handler.postDelayed (task, interval);
    }

    public void stopTimer () {
        paused = true;
    }

    public Timer (Runnable runnable, int interval, boolean started) {
        handler = new Handler ();
        this.runnable = runnable;
        this.interval = interval;
        if (started)
            startTimer ();
    }
}

关于android - 如何处理 Timer was cancelled 错误?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36636989/

相关文章:

android - 如何在Android中测试基于Espresso的Intent?

android - 如何使用 Mockito 或 MockK 模拟 android.util.Patterns

android - 在intellij idea上设置android wear UI库

android - 如何确定设备是否有振动器?

java - 检查并读取 android 中主 json 数组中的内部 json 数组?

Android - 在运行时更改自定义标题 View

android - 如何在android ..中使用淡入淡出到背景图像?

java - OpenGL ES 2.0 : From orthogonal to perspective (card flip effect)

android - 安卓人脸识别?

java - 是否有用于 GCM 云连接服务器的 Java API