处理程序postDelayed()导致意外变量变化

时间:2014-11-24 06:51:33

标签: android multithreading postdelayed

我正在开发一款游戏,我正在使用postDelay()让动画更流畅。我遇到的一个问题是,如果我在其中执行postDelay()的方法,那么该方法操作的变量不会以意想不到的方式发生变化。这会导致奇怪的行为,例如位图无法正确绘制。

以下是麻烦的部分:

       synchronized private void rotate(int dir) {
            final int originalAngle = angle; //angle is the suspicious variable
            final int steps = 4;

            for (int i = 0; i != steps; i++) {
                smoothAnimateRotate(i, steps, originalAngle, dir);
            }
            finishAnimateRotate(steps);
        }

        private void smoothAnimateRotate(final int i, final int steps, final int originalAngle, final int dir) {
            handler.postDelayed(new Runnable() {
                @Override
                public void run() {
                    angle = originalAngle + dir*(90*(i+1))/steps;
                    rotateStep();
                }
            }, 100 * (i + 1));
        }


        private void rotateStep() {
            angle = (angle+1800) % 360;
            viewdx = (int) (Math.cos(radify(angle))*(1<<16));
            viewdy = (int) (Math.sin(radify(angle))*(1<<16));
            moveStep();
        }

        private void moveStep() {
            notifyViewerRedraw() ;
            try {
                Thread.currentThread().sleep(25);
            } catch (Exception e) { }
        }

rotate()被非常快速地调用时,我认为变量angle不知何故搞砸了。当我将rotate()大约相隔一秒钟时,这种情况不会发生,但如果我这么做很快,angle只会被假定为0,90,180或270到45度或其他东西。我无法弄清楚为什么和调试没有提供丰硕的成果。那么如何确保angle以这种意想不到的方式进行修改?

1 个答案:

答案 0 :(得分:0)

我修好了。我在Runnables中放了一个名为isRunning的布尔值并打开和关闭它。

synchronized private void rotate(int dir) {
    final int originalAngle = angle;
    final int steps = 4;

    if (!isRunning) {
        for (int i = 0; i != steps; i++) {
            smoothAnimateRotate(i, steps, originalAngle, dir);
        }
        finishAnimateRotate(steps);
    }
}

private void smoothAnimateRotate(final int i, final int steps, final int originalAngle, final int dir) {
    handler.postDelayed(new Runnable() {
        @Override
        public void run() {
            isRunning = true;
            angle = originalAngle + dir*(90*(i+1))/steps;
            rotateStep();
        }
    }, 100 * (i + 1));
}

private void finishAnimateRotate(final int steps) {
    handler.postDelayed(new Runnable() {
        @Override
        public void run() {
            rotateFinish();
            isRunning = false;
        }
    }, 100 * (steps + 1));
}