最近在开发一个美颜的相机功能,需要自定义抓取相机回调的数,生成照片并保存到相册,需要自定义保存照片。
相机开始使用时没有任何问题,测试拍照几次后突然奔溃,跟踪代码日志发现是图片保存失败,同样的代码,同样的逻辑,正常使用很多次以后才奔溃报错,根据日志找到报错的位置,
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源码仓库 可以看到下载方式,这个是官方公布的下载方式哦。
标签:java,name,自定义,image,MediaProvider,Android,com,bug,android From: https://blog.csdn.net/m0_67228868/article/details/140957260