6

我正在为 Android 2.2 设计计时器/倒数计时器应用程序,并希望按下一个按钮即可同时启动计时器计时器。因此,理想情况下,我希望计时器和计时器上的秒数(时间)同时更改。(即使计时器正在向上计数,计时器也会倒计时)。由于我使用的是 Android 提供的计时器和计时器功能,因此当用户按下“开始”按钮时,我编写了以下代码

private boolean mStartPressedOnce = false;
long mTimeWhenStopped = 0;
Chronometer mChronometer;   
MyCounter mCounter;
...
@Override
public void onClick(View v) {
    switch (v.getId()) {
    case R.id.StartButton:
        // Perform some initialization for the chronometer depending
        // on the button press
        if (mStartPressedOnce == false) {
            mChronometer.setBase(SystemClock.elapsedRealtime());
        } else {
            mChronometer.setBase(SystemClock.elapsedRealtime() + mTimeWhenStopped);
        }

        // Perform the initialization for the timer
        mCounter = new MyCount(45000, 1000);

        // Fire up the chronometer  
        mChronometer.start();

        // Fire up the timer
        mCounter.start();
        break;

        case R.id.ResetButton:
            // Reset the chronometer  
            mChronometer.setBase(SystemClock.elapsedRealtime());
            mTimeWhenStopped = 0;
            break;

        case case R.id.StopButton:
            mStartPressedOnce = true;
            // Stop the chronometer 
            mTimeWhenStopped = mChronometer.getBase() - SystemClock.elapsedRealtime();
            mChronometer.stop();
            break;
    }

... 

public class MyCounter extends CountDownTimer {

    @Override
    public MyCount(long millisInFuture, long countDownInterval) {
        super(millisInFuture, countDownInterval);
    }

    @Override
    public void onFinish() {        
        // Nothing to do here
    }

    @Override
    public void onTick(long millisUntilFinished) {
        long seconds = (long) (millisUntilFinished / 1000);
        long minutes = (long) ((millisUntilFinished / 1000) / 60);
        long hours = (long) (((millisUntilFinished / 1000) / 60) / 60);

        // Do some formatting to make seconds, minutes and hours look pretty    

        // Update the timer TextView            
       (TextView) findViewById(R.id.CountDownTimerTextView))
           .setText(hours + ":" + minutes + ":" + seconds);
    }
}

虽然看起来计时器和计时器上的秒数最初是同步的,但在很短的时间之后,它们似乎会消失,并且两者的第二次更新发生在不同的时间。

想知道我能做些什么来解决这个问题。我确实遇到过 - 并阅读了这个帖子

同时运行多个 AsyncTask —— 不可能吗?

我意识到可能需要进行设计更改,但我不确定需要做什么。

编辑:包括计时器和计时器的类型以及使用 Chronometer 计算时间的方法 - 根据 jolivier 和 njzk2 的建议

4

2 回答 2

2

您可以使用 System.currentTimeMillis() 检索当前时间,将其存储到变量中并将其转发给两者mChronometermCounter,以便它们使用相同的时间参考,尽管它们的任务在不同的时间开始。

编辑:对于给定的类型,关于 Chronometer 的 android 文档会告诉你,你可以使用elapsedRealTime来实现我所说的。 CountDownTimer没有这个,它的start方法是最终的,所以你可能想要使用另一个实现,更好地了解你的用例可能会对我们有所帮助。

基本上,希望两个线程在同一毫秒执行一个动作绝不是一个好主意,其中一个将用作时钟,另一个必须是从属并监听时钟。

于 2012-11-20T12:51:27.990 回答
2

因此,在考虑了一段时间并放弃了jolivier如此慷慨地与我们分享的建议之后,我意识到存在一个名为的方法,该方法onChronometerTick在每次有计时器滴答声时(在这种情况下是每秒)都会被调用。所以,我想每次调用该方法时从计数器中减去 1000 毫秒,并相应地更新计时器显示。我完全摆脱了 Android 计时器 ( CountDownTimer)。我认为这将是同时更新两个显示器的好方法。它也是一个简单的定时器实现。

我很高兴地报告它似乎运作良好。计时器和计时器显示确实同时更新。所以,原来的问题看起来已经回答了。不幸的是,我在计时器前面遇到了一个错误的错误,我用一个丑陋的黑客修复了这个错误。我正在发布到目前为止的内容。欢迎任何有关如何修复黑客或改进代码的建议。请注意,我已对代码进行了注释,以尝试使其易于理解所做的工作。

错误编辑:我注意到的另一件事是,大约 10 分钟左右后,计时器和计时器关闭了一秒钟。更准确地说,计时器比计时器落后一秒。尚不确定为什么会发生这种情况。

private boolean mStartPressedOnce = false;
long mTimeWhenStopped = 0;
Chronometer mChronometer;   
long millisUntilFinished = 0;
boolean firstPassOver = false;
int counter = 0;
...
@Override
public void onClick(View v) {
    switch (v.getId()) {
    case R.id.StartButton:
        // Perform some initialization for the chronometer depending
        // on the button press
        if (mStartPressedOnce == false) {
            mChronometer.setBase(SystemClock.elapsedRealtime());
        } else {
            mChronometer.setBase(SystemClock.elapsedRealtime() + mTimeWhenStopped);
        }

        // Fire up the chronometer  
        mChronometer.start();
        break;

        case R.id.ResetButton:
            // Reset the chronometer  
            mChronometer.setBase(SystemClock.elapsedRealtime());
            mTimeWhenStopped = 0;
            break;

        case case R.id.StopButton:
            mStartPressedOnce = true;
            // Stop the chronometer 
            mTimeWhenStopped = mChronometer.getBase() - SystemClock.elapsedRealtime();
            mChronometer.stop();
            break;
    }

... 

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.stop_watch);

    mChronometer = (Chronometer) findViewById(R.id.StopWatchTextView);

    // Initialize the number of milliseconds before the timer expires (
    // set the timer) - in this case to 46 seconds
    millisUntilFinished = 46000;

    // Display how many seconds remain before the timer expires
    ((TextView) findViewById(R.id.CountDownTimerTextView)).setText(hours
            + ":" + minutes + ":" + millisUntilFinished / 1000);

    // In line with the suggestion provided by  jolivier - make the timer
    // the slave and  update its display every time the chronometer 
    // ticks        
    mChronometer
            .setOnChronometerTickListener(new Chronometer.OnChronometerTickListener() {

                @Override
                public void onChronometerTick(Chronometer chronometer) {

                    // Update the display for the chronometer
                    CharSequence text = chronometer.getText();                  
                    chronometer.setText(text);

                    // Update the display for the timer                 
                    // !!! BUG !!!
                    // Looks like onChronometerTick is called at the 0th second
                    // and this causes an off by two error if a count down timer
                    // is being implemented. Fixed it with this hack. There's gotta
                    // be a more elegant solution, though.
                    if(counter >= 2) {
                        millisUntilFinished1 = millisUntilFinished1 - 1000;
                        counter = 2;
                    }
                    counter++;

                    if (millisUntilFinished >= 0) {

                        long seconds = (long) (millisUntilFinished / 1000);
                        long minutes = (long) ((millisUntilFinished / 1000) / 60);
                        long hours = (long) (((millisUntilFinished / 1000) / 60) / 60);

                        // Do some formatting to make seconds, minutes and hours look pretty    

                        // Update the timer TextView     
                        ((TextView) findViewById(R.id.CountDownTimerTextView))
                                .setText(hours + ":" + minutes + ":"
                                        + seconds);
                        }
                    }

    });
    // Other code
    ...
}
于 2012-11-21T05:50:41.223 回答