当屏幕方向改变时,lockCanvas失败

3

我正在尝试在Android中构建一个动态壁纸,但当屏幕方向改变时,应用程序会崩溃。看起来似乎在尝试锁定Surface Holder上的Canvas时崩溃了,但我不确定该怎么做才能防止这种情况发生。

这是类:

public class LiveWallpaperService extends WallpaperService
{
    public void onCreate() {
        super.onCreate();
    }

    public void onDestroy() {
        super.onDestroy();
    }

    public Engine onCreateEngine() {
        return new MyWallpaperEngine();
    }

    class MyWallpaperEngine extends Engine
    {
        private final Handler handler = new Handler();
        private final Runnable drawRunner = new Runnable() {
            @Override
            public void run() {
                draw();
            }
        };
        private boolean visible = true;

        Paint paint;

        MyWallpaperEngine() {
            paint = new Paint();
        }

        public void onCreate(SurfaceHolder surfaceHolder) {
            super.onCreate(surfaceHolder);
        }

        @Override
        public void onVisibilityChanged(boolean visible) {
            this.visible = visible;

            if (visible) {
                handler.post(drawRunner);
            }
            else {
                handler.removeCallbacks(drawRunner);
            }
        }

        @Override
        public void onSurfaceDestroyed(SurfaceHolder holder) {
            super.onSurfaceDestroyed(holder);
            this.visible = false;
            handler.removeCallbacks(drawRunner);
        }

        public void onOffsetsChanged(float xOffset, float yOffset, float xStep, float yStep, int xPixels, int yPixels) {
            draw();
        }

        void draw() {
            final SurfaceHolder holder = getSurfaceHolder();

            Canvas c = null;

            try {
                c = holder.lockCanvas();

                if (c != null) {
                    // Paint stuff here.
                }
            }
            finally {
                if (c != null) {
                    holder.unlockCanvasAndPost(c);
                }
            }

            handler.removeCallbacks(drawRunner);
            if (visible) {
                handler.postDelayed(drawRunner, 10);
            }
        }
    }
}

当屏幕方向改变时,会发生以下异常:

E/StudioProfiler: JVMTI error: 15(JVMTI_ERROR_THREAD_NOT_ALIVE) 
E/Surface: dequeueBuffer failed (No such device)
E/BaseSurfaceHolder: Exception locking surface
   java.lang.IllegalArgumentException
           at android.view.Surface.nativeLockCanvas(Native Method)
           at android.view.Surface.lockCanvas(Surface.java:318)
           at com.android.internal.view.BaseSurfaceHolder.internalLockCanvas(BaseSurfaceHolder.java:194)
           at com.android.internal.view.BaseSurfaceHolder.lockCanvas(BaseSurfaceHolder.java:158)
           at android.service.wallpaper.WallpaperService$Engine$1.lockCanvas(WallpaperService.java:262)
           at greencell.bitpatternswallpaper.LiveWallpaperService$MyWallpaperEngine.draw(LiveWallpaperService.java:206)
           at greencell.bitpatternswallpaper.LiveWallpaperService$MyWallpaperEngine$1.run(LiveWallpaperService.java:51)
           at android.os.Handler.handleCallback(Handler.java:790)
           at android.os.Handler.dispatchMessage(Handler.java:99)
           at android.os.Looper.loop(Looper.java:164)
           at android.app.ActivityThread.main(ActivityThread.java:6494)
           at java.lang.reflect.Method.invoke(Native Method)
           at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:438)
           at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:807)

更新:

我查看了许多其他帖子,似乎都存在相同的问题,但到目前为止,我唯一能做的就是将unlockCanvasAndPostlockCanvas包装在try catch中以忽略IllegalArgumentException


1
我不明白。那个链接中的问题与这里发生的完全不同。 - Green Cell
我尝试使用您提供的代码重现问题,但是我无法做到。它是否发生在特定的Android版本/设备上? - Marcin Jedynak
现在我正在使用 Android 8.1.0 版本的 Pixel 2 进行测试。 - Green Cell
2个回答

2
draw()中,我建议将handler.removeCallbacks(drawRunner);放在try块之前。可能是因为在方向更改时调用了onOffsetsChanged(),而处理程序上的先前线程可能尚未调用unlockCanvasAndPost(c),这就解释了为什么此时使用lockCanvas()会出现错误。但是,如果您发布的代码与本地运行的代码完全相同,则不应该出现这种情况,因为您没有覆盖onOffsetsChanged()
另一件可以尝试的事情是重写onSurfaceChanged()并清除处理程序队列,如下所示:
@Override
public void onSurfaceChanged(SurfaceHolder holder, int format, int width, int height) {
    handler.removeCallbacks(drawRunner);
    super.onSurfaceChanged(holder, format, width, height);
}

最终,我在网上阅读的有关WallpaperService的所有示例都有一个try-finally块,其中包含锁定/解锁画布逻辑,因此我不必担心。


错误偶尔仍会触发,但感觉更加稳定了。是的,我注意到所有的示例最终都以try catch结束,感觉有点不好,但我想我必须接受它:)。谢谢! - Green Cell
这很奇怪,但我很高兴能够帮忙! :) - Cvarier

2
在清单文件中的<service android:name=".LiveWallpaperService" />添加android:configChanges="orientation",例如:<service android:name=".LiveWallpaperService" android:configChanges="orientation" />

谢谢您的回答,尽管我没有注意到任何区别。 - Green Cell

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接