记一次 Android 自定义相机拍照奔溃bug事件

最近在开发一个美颜的相机功能,需要自定义抓取相机回调的数,生成照片并保存到相册,需要自定义保存照片。

相机开始使用时没有任何问题,测试拍照几次后突然奔溃,跟踪代码日志发现是图片保存失败,同样的代码,同样的逻辑,正常使用很多次以后才奔溃报错,根据日志找到报错的位置,

ContentValues values = new ContentValues();
values.put(MediaStore.Images.Media.DISPLAY_NAME, "image_name");
values.put(MediaStore.Images.Media.MIME_TYPE, "image/jpeg");
Uri uri = context.getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);

在insert这一行发生的错误,表面看没有任何问题,代码一切正常。

报错日志如下

Writing exception to parcel
java.lang.IllegalStateException: Failed to build unique file: /storage/emulated/0/Pictures/image_name owner_package_name=com.vedavision.gockr _display_name=image_name mime_type=image/jpeg _data=/storage/emulated/0/Pictures/image_name relative_path=Pictures/
at com.android.providers.media.MediaProvider.ensureFileColumns(MediaProvider.java:3933)
At com.android.providers.media.MediaProvider.ensureUniqueFileColumns(MediaProvider.java:3650)
at com.android.providers.media.MediaProvider.insertFile(MediaProvider.java:4423)
at com.android.providers.media.MediaProvider.insertInternal(MediaProvider.java:4890)
at com.android.providers.media.MediaProvider.insert(MediaProvider.java:4709)
at android.content.ContentProvider$Transport.insert(ContentProvider.java:348)
at android.content.ContentProviderNative.onTransact(ContentProviderNative.java:169)
at android.os.Binder.execTransactInternal(Binder.java:1285)
at android.os.Binder.execTransact(Binder.java:1249)

具体报错位置提示:

java.lang.IllegalStateException: Failed to build unique file: /storage/emulated/0/Pictures/image_name owner_package_name=com.vedavision.gockr _display_name=image_name mime_type=image/jpeg _data=/storage/emulated/0/Pictures/image_name relative_path=Pictures/
at android.os.Parcel.createExceptionOrNull(Parcel.java:3019)
at android.os.Parcel.createException(Parcel.java:2995)
at android.os.Parcel.readException(Parcel.java:2978)
at android.database.DatabaseUtils.readExceptionFromParcel(DatabaseUtils.java:190)
at android.database.DatabaseUtils.readExceptionFromParcel(DatabaseUtils.java:142)
at android.content.ContentProviderProxy.insert(ContentProviderNative.java:565)
at android.content.ContentResolver.insert(ContentResolver.java:2196)
at android.content.ContentResolver.insert(ContentResolver.java:2157)
at com.vedavision.gockr.utils.ImageUtils.saveToGallery(ImageUtils.java:181)
at com.vedavision.gockr.camera.FrameDrawer.draw(FrameDrawer.java:249)
at com.vedavision.gockr.camera.FrameDrawer.draw(FrameDrawer.java:195)
at com.vedavision.gockr.camera.SimpleCameraRenderer.onDrawFrame(SimpleCameraRenderer.java:213)
at android.opengl.GLSurfaceView$GLThread.guardedRun(GLSurfaceView.java:1590)
at android.opengl.GLSurfaceView$GLThread.run(GLSurfaceView.java:1272)

从日志看 最终报错的位置是:

com.android.providers.media.MediaProvider.ensureFileColumns的方法,如下

这里可以看到我们日志中的错误,Failed to build unique file,但是明显这个错误是FileUtils.buildNonUniqueFile  或者 buildUniqueFile抛出的,我们继续跟踪,进入到buildUniqueFile方法,  在这个里面调用了 FileUtils.buildUniqueFileWithExtension(),

buildUniqueFileWithExtension这个方法注意看 n++>=32抛出异常,大概意思就是,如果我们以固定名称创建文件,不是覆盖原来的文件,而是在文件的后面增加(0),(1)这样的数字,这里的判断就相当于同一个文件名称我们最多不可以超过32,既然问题找到了,那就修复,既然不让文件名重复,那我们就在赋值文件名的地方改成随机生成文件名,我这里采用的是UUID的方式。


values.put(MediaStore.Images.Media.DISPLAY_NAME, UUID.randomUUID().toString());

这样就可以解决问题了。

这种底层限制不出问题,一般情况不会去关注,但是出了问题要学会怎么找问题,怎么修正错误。

com.android.providers.media.MediaProvider 这个类是android底层源码,需要下载源码才可以看到,直接跟代码是找不到这个类的,这点需要注意,源码获取的方式在 android源码仓库 可以看到下载方式,这个是官方公布的下载方式哦。

  • 8
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值