0

我向这个处理程序发送消息(这里我省略了一些情况)对位图进行一些操作,然后在 UI 上设置位图。因为对位图的操作比较耗时,所以我把它放在了一个线程中。完成此操作后,我使用runOnUiThread方法在 UI 线程中设置 UI 。
通常,这一切都运行良好,但在某些情况下,我得到了以下异常:java.lang.RuntimeException: Canvas: 试图使用回收的位图 android.graphics.Bitmap
这种情况非常罕见,但确实发生了好几次。

private Handler mHandler = new Handler()
{
    @Override
    public void handleMessage(Message msg)
    {
        int what = msg.what;
        switch (what)
        {

            case VideoProtocal.COMPRESS_SUCCESS:
                state = State.COMPRESSED;
                recordBack.setVisibility(View.VISIBLE);
                recordSend.setVisibility(View.VISIBLE);
                playButton.setVisibility(View.VISIBLE);
                new Thread(){
                    @Override
                    public void run()
                    {
                        rotationBitmap();
                        runOnUiThread(new Runnable() {
                            public void run() {
                                if( null != bitmap && !bitmap.isRecycled() )
                                {
                                    Drawable drawable = new BitmapDrawable(bitmap);
                                    fristFrame.setBackgroundDrawable(drawable);
                                }
                            }
                        });
                    }
                }.start();
                mFrameCount = 0;
                getVideoSize();
                videoTime = getVideoFileTime(mp4_path) / 1000;
                timeView.setText(getString(R.string.video_count, videoTime));
                deleteTempFile();
                LogX.trace(TAG, "COMPRESS_SUCCESS");
                dismissProgress();
                break;
            default:
                break;
        }
    }

};

方法rotationBitmap() 和它调用的一些方法如下所示:

private void rotationBitmap()
{
    int width = (int) (PIC_WIDTH * FusionField.currentDensity);
    int height = (int) (PIC_HIGHT * FusionField.currentDensity);
    Bitmap bmTemp = bitmap;
    bitmap = CommonUtil.resizeImage(bitmap, width, height);

    if (bmTemp != null && !bmTemp.isRecycled() && bmTemp != bitmap)
    {
        bmTemp.recycle();
        bmTemp = null;
    }

    rotate(bitmap, rotationAngle());

    CommonUtil.checkMysoftStage(FusionCode.videoThumbnail);
    String videoName = CommonUtil.getUniqueName(".jpg", FusionCode.videoName);
    jpg_path = FusionCode.videoThumbnail + "/" + videoName;
    try
    {
        FileOutputStream fos = new FileOutputStream(jpg_path);
        if (bitmap != null)
        {
            bitmap.compress(CompressFormat.JPEG, 50, fos);
        }
        fos.close();
    }
    catch (FileNotFoundException e)
    {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
    catch (IOException e)
    {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}


private Bitmap rotate(Bitmap b, int degrees)
{
    if (degrees != 0 && b != null)
    {
        Matrix m = new Matrix();
        m.setRotate(degrees, (float) b.getWidth() / 2, (float) b.getHeight() / 2);
        try
        {
            bitmap = Bitmap.createBitmap(b, 0, 0, b.getWidth(), b.getHeight(), m, true);
        }
        catch (OutOfMemoryError ex)
        {
            // TODO Auto-generated catch block
            ex.printStackTrace();
        }
    }
    return b;
}

从代码中,问题必须出现在此语句中:

Drawable drawable = new BitmapDrawable(bitmap);

但是根据代码上下文,位图不应该为空或回收,这真的很奇怪。

我一直在这个问题上徘徊近两天,无法弄清楚问题是什么,但问题确实存在。有人可以给我一些建议吗?任何回应表示赞赏!谢谢!<br />

顺便说一句,当我尝试这样的 RuntimeException 时,无法捕获这种异常,为什么?

try
{
    bitmap.recycle();
    Drawable drawable = new BitmapDrawable(bitmap);
    fristFrame.setBackgroundDrawable(drawable);
}
catch (RuntimeException e)
{
    e.printStackTrace();
}
4

1 回答 1

0

好的,让我回答我的问题。出现此问题的原因是 VideoProtocal.COMPRESS_SUCCESS 被 mHandler 接收了两次。它在那里有多线程,所以在某些时候,会出现“使用回收位图”的问题。至于为什么收到两次消息,呃......这是我使用定时器任务的错误。

于 2012-08-01T01:40:09.470 回答