カスタム SurfaceView を使用して小さなゲームを作成しています。SurfaceView の下部に AdMob ビューを配置したかったのですが、アプリを約 6 分間実行した後、アプリがエラーをスローして電話を再起動しました。AdMob ビューを 1 つのボタンに置き換えましたが、引き続きエラーが発生するため、SurfaceView の上にビューを配置したことが問題の原因であると考えられます。SurfaceView の上からビューを削除すると、ゲームは無期限に正常に実行されます。
RelativeLayout 以外のレイアウトを使用してみましたが、問題は解決しません。誰かが以前にこれに遭遇したことがあるか、それとも他に探索する道があるかどうか疑問に思っています.
カスタム SurfaceView:
public class AndroidFastRenderView extends SurfaceView implements Runnable
{
private static final String TAG = "AndroidFastRenderView";
AndroidGame game;
Bitmap framebuffer;
Thread renderThread = null;
SurfaceHolder holder;
volatile boolean running = false;
public AndroidFastRenderView(AndroidGame game, Bitmap framebuffer)
{
super(game);
this.game = game;
this.framebuffer = framebuffer;
this.holder = getHolder();
}
public AndroidFastRenderView(Context context)
{
super(context);
}
public void resume()
{
running = true;
renderThread = new Thread(this);
renderThread.start();
}
public void run()
{
Rect dstRect = new Rect(0,0,80,120);
long startTime = System.nanoTime();
while (running)
{
if(!holder.getSurface().isValid())
{
continue;
}
float deltaTime = (System.nanoTime() - startTime) / 1000000000.0f;
startTime = System.nanoTime();
game.getCurrentScreen().update(deltaTime);
game.getCurrentScreen().present(deltaTime);
Canvas canvas = null;
canvas = holder.lockCanvas();
if (canvas != null)
{
dstRect = canvas.getClipBounds();
canvas.drawBitmap(framebuffer, null, dstRect, null);
holder.unlockCanvasAndPost(canvas);
}
}
}
public void pause()
{
Log.e(TAG, "Paused");
running = false;
while(true)
{
try {
renderThread.join();
break;
} catch (InterruptedException e) {
// retry
}
}
}
}
メイン アクティビティでのインターフェイスの構築:
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
boolean isLandscape = getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE;
int frameBufferWidth = isLandscape ? 120 : 80;
int frameBufferHeight = isLandscape ? 80 : 120;
Bitmap frameBuffer = Bitmap.createBitmap(frameBufferWidth, frameBufferHeight, Config.RGB_565);
float scaleX = (float) frameBufferWidth / getWindowManager().getDefaultDisplay().getWidth();
float scaleY = (float) frameBufferHeight / getWindowManager().getDefaultDisplay().getHeight();
renderView = new AndroidFastRenderView(this, frameBuffer);
graphics = new AndroidGraphics(getAssets(), frameBuffer);
fileIO = new AndroidFileIO(getAssets());
audio = new AndroidAudio(this);
input = new AndroidInput(this, renderView, scaleX, scaleY);
screen = this.getStartScreen();
PowerManager powerManager = (PowerManager) getSystemService(Context.POWER_SERVICE);
wakeLock = powerManager.newWakeLock(PowerManager.FULL_WAKE_LOCK, "GLGame");
if (useAds)
{
//adView = new AdView(this, AdSize.SMART_BANNER, AD_UNIT_ID);
LayoutParams params = new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
params.addRule(RelativeLayout.ALIGN_PARENT_BOTTOM);
//adView.setLayoutParams(params);
RelativeLayout layout = new RelativeLayout(this);
layout.addView(renderView);
//layout.addView(adView);
Button button = new Button(this);
button.setLayoutParams(params);
button.setText("Blargle");
layout.addView(button);
//adView.loadAd(new AdRequest());
setContentView(layout);
}
else
{
setContentView(renderView);
}
}
エラーの LogCat:
01-25 13:50:18.084: E/SurfaceTextureClient(10342): dequeueBuffer failed (Broken pipe)
01-25 13:50:18.084: E/SurfaceHolder(10342): Exception locking surface
01-25 13:50:18.084: E/SurfaceHolder(10342): java.lang.IllegalArgumentException
01-25 13:50:18.084: E/SurfaceHolder(10342): at android.view.Surface.lockCanvasNative(Native Method)
01-25 13:50:18.084: E/SurfaceHolder(10342): at android.view.Surface.lockCanvas(Surface.java:88)
01-25 13:50:18.084: E/SurfaceHolder(10342): at android.view.SurfaceView$4.internalLockCanvas(SurfaceView.java:754)
01-25 13:50:18.084: E/SurfaceHolder(10342): at android.view.SurfaceView$4.lockCanvas(SurfaceView.java:734)
01-25 13:50:18.084: E/SurfaceHolder(10342): at com.somethingutility.games.framework.impl.AndroidFastRenderView.run(AndroidFastRenderView.java:60)
01-25 13:50:18.084: E/SurfaceHolder(10342): at java.lang.Thread.run(Thread.java:856)
このエラーは、電話が再起動するまで繰り返されます。