Android: CountDownTimer 在Tick()上最后跳过!

2022-08-31 20:11:19

法典:

public class SMH extends Activity {  

    public void onCreate(Bundle b) {  
        super.onCreate(b);  
        setContentView(R.layout.main);  

        TextView tv = (TextView) findViewById(R.id.tv);  

        new CountDownTimer(10000, 2000) {  
            public void onTick(long m) {  
               long sec = m/1000+1;  
               tv.append(sec+" seconds remain\n");  
            }  
            public void onFinish() {  
               tv.append("Done!");  
            }  
        }.start();  
   }

输出:
剩余
10 秒 8 秒剩余
6 秒剩余
4 秒剩余
完成!

问题:

如何让它显示“剩余 2 秒”?经过的时间确实是10秒,但最后一次onTick()从未发生过。如果我将第二个参数从 2000 更改为 1000,则输出如下:

10 秒剩余
9 秒剩余
8 秒剩余
7 秒剩余
6 秒剩余
5 秒剩余
4 秒剩余
3 秒剩余
2 秒剩余
完成!

所以你看,它似乎跳过了最后一个onTick()调用。顺便说一句,XML文件基本上是默认的主.xml TextView分配了id tv,文本设置为“”。


答案 1

我检查了CountDownTimer的源代码。“缺失的勾选”来自CountDownTimer的一个特殊功能,我还没有在其他地方看到过:

在每次价格变动开始时,在调用 onTick() 之前,计算倒计时结束前的剩余时间。如果此时间小于倒计时时间间隔,则不再调用 onTick。相反,只安排下一个价格变动(将调用 onFinish() 方法)。

鉴于硬件时钟并不总是非常精确,后台可能还有其他进程延迟运行CountDownTimer的线程,并且Android本身在调用CountDownTimer的消息处理程序时可能会产生一个小的延迟,因此在倒计时结束之前对最后一次刻度的调用很可能至少延迟一毫秒,因此onTick()不会叫。

对于我的应用程序,我通过使刻度间隔“略微”变小(500毫秒)来解决此问题。

    myCountDownTimer = new CountDownTimer(countDownTime, intervalTime - 500) {
                                   ...
    }

我可以保持我的代码不变。对于间隔时间长度至关重要的应用程序,此处发布的其他解决方案可能是最好的。


答案 2

我不知道为什么最后一个价格变动不起作用,但您可以使用 Runable 创建自己的计时器,例如。

class MyCountDownTimer {
    private long millisInFuture;
    private long countDownInterval;
    public MyCountDownTimer(long pMillisInFuture, long pCountDownInterval) {
            this.millisInFuture = pMillisInFuture;
            this.countDownInterval = pCountDownInterval;
        }
    public void Start() 
    {
        final Handler handler = new Handler();
        Log.v("status", "starting");
        final Runnable counter = new Runnable(){

            public void run(){
                if(millisInFuture <= 0) {
                    Log.v("status", "done");
                } else {
                    long sec = millisInFuture/1000;
                    Log.v("status", Long.toString(sec) + " seconds remain");
                    millisInFuture -= countDownInterval;
                    handler.postDelayed(this, countDownInterval);
                }
            }
        };

        handler.postDelayed(counter, countDownInterval);
    }
}

并开始它,

new MyCountDownTimer(10000, 2000).Start();

编辑高飞的问题

你应该有一个变量来保持计数器状态(布尔值)。然后你可以写一个Stop()方法,如Start()。

编辑-2 对于高飞的问题

实际上停止计数器上没有错误,但是在停止(恢复)后再次启动时有错误。

我正在编写一个新的更新的完整代码,我刚刚尝试过,它正在工作。这是一个基本的计数器,通过开始和停止按钮在屏幕上显示时间。

计数器的类

public class MyCountDownTimer {
    private long millisInFuture;
    private long countDownInterval;
    private boolean status;
    public MyCountDownTimer(long pMillisInFuture, long pCountDownInterval) {
            this.millisInFuture = pMillisInFuture;
            this.countDownInterval = pCountDownInterval;
            status = false;
            Initialize();
    }

    public void Stop() {
        status = false;
    }

    public long getCurrentTime() {
        return millisInFuture;
    }

    public void Start() {
        status = true;
    }
    public void Initialize() 
    {
        final Handler handler = new Handler();
        Log.v("status", "starting");
        final Runnable counter = new Runnable(){

            public void run(){
                long sec = millisInFuture/1000;
                if(status) {
                    if(millisInFuture <= 0) {
                        Log.v("status", "done");
                    } else {
                        Log.v("status", Long.toString(sec) + " seconds remain");
                        millisInFuture -= countDownInterval;
                        handler.postDelayed(this, countDownInterval);
                    }
                } else {
                    Log.v("status", Long.toString(sec) + " seconds remain and timer has stopped!");
                    handler.postDelayed(this, countDownInterval);
                }
            }
        };

        handler.postDelayed(counter, countDownInterval);
    }
}

活动类

public class CounterActivity extends Activity {
    /** Called when the activity is first created. */
    TextView timeText;
    Button startBut;
    Button stopBut;
    MyCountDownTimer mycounter;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        timeText = (TextView) findViewById(R.id.time);
        startBut = (Button) findViewById(R.id.start);
        stopBut = (Button) findViewById(R.id.stop);
        mycounter = new MyCountDownTimer(20000, 1000);
        RefreshTimer();
    }

    public void StartTimer(View v) {
        Log.v("startbutton", "saymaya basladi");
        mycounter.Start();
    }

    public void StopTimer(View v) {
        Log.v("stopbutton", "durdu");
        mycounter.Stop();
    }

    public void RefreshTimer() 
    {
        final Handler handler = new Handler();
        final Runnable counter = new Runnable(){

            public void run(){
                timeText.setText(Long.toString(mycounter.getCurrentTime()));
                handler.postDelayed(this, 100);
            }
        };

        handler.postDelayed(counter, 100);
    }
}

主要.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:weightSum="1">
    <TextView android:textAppearance="?android:attr/textAppearanceLarge" 
              android:text="TextView" android:layout_height="wrap_content" 
              android:layout_width="wrap_content" 
              android:id="@+id/time">
    </TextView>
    <Button android:text="Start" 
            android:id="@+id/start" 
            android:layout_width="wrap_content" 
            android:layout_height="wrap_content" 
            android:onClick="StartTimer">
    </Button>
    <Button android:text="Stop" 
            android:id="@+id/stop" 
            android:layout_width="wrap_content" 
            android:layout_height="wrap_content" 
            android:onClick="StopTimer">
    </Button>
</LinearLayout>