targetSdk27 FileProvider 摄像和照相[通俗易懂]

大家好,又见面了,我是你们的朋友全君。

今天我们来讨论一下在android 7.0及以上版本中,如何解决相机和摄像头在严格模式下的问题。以下是详细的解决方案和配置步骤。

官方推荐 – FileProvider

1. 缘由(转载)

从Android 7.0(N)开始,系统会严格执行StrictMode模式,这意味着会对安全性进行更严格的检查。从Android N开始,不允许在应用间使用file://的方式传递文件,否则会抛出FileUriExposedException错误,直接导致应用崩溃。

然而,官方提供了解决方案,即使用FileProvider,通过content://模式替换file://,需要开发者将targetSdkVersion升级到24及以上才会执行此策略。

2. 配置 FileProvider

2.1 AndroidManifest内配置FileProvider

注意:provider是Android四大组件之一,需要放置在application标签内。

<provider     android:authorities="${applicationId}.fileprovider"     android:exported="false"     android:grantUriPermissions="true"     android:name="android.support.v4.content.FileProvider">     <meta-data         android:name="android.support.FILE_PROVIDER_PATHS"         android:resource="@xml/install"/> </provider>

2.2 可访问路径配置

路径配置的位置在install.xml文件中:

<?xml version="1.0" encoding="utf-8"?> <paths xmlns:android="http://schemas.android.com/apk/res/android">     <external-path name="DCIM" path="DCIM/camerademo"/>     <external-path name="external_storage_root" path="."/>     <external-path name="Pictures" path="Pictures/camerademo"/>     <files-path name="private_files" path="images"/>     <cache-path name="private_cache" path="images"/>     <external-files-path name="external_files" path="Pictures"/>     <external-cache-path name="external_cache" path="."/> </paths>

分析路径的用途

当调用Environment.getExternalStorageDirectory()获取外部存储根目录时,获取的路径是/storage/emulated/0。

例如,在代码中使用photoUri = FileProvider.getUriForFile(this, getPackageName() + “.fileprovider”, takeImageFile);时,photoUri的值可能是:

content://com.yoshin.company.blogdemo.fileprovider/external_storage_root/DCIM/camera/IMG_20200526_203123.jpg

此时,外部存储根目录在Uri中显示为“external_storage_root”。

其他路径配置的标签参照FileProvider里面的TAG配置

targetSdk27 FileProvider 摄像和照相[通俗易懂]

3. 访问 FileProvider

调用相机的代码可以如下修改:

public void takePicture(Activity activity, int requestCode) {     Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);     intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION | Intent.FLAG_ACTIVITY_CLEAR_TOP);     if (existSDCard()) {         takeImageFile = new File(Environment.getExternalStorageDirectory(), "/DCIM/camera/");     } else {         takeImageFile = Environment.getDataDirectory();     }     takeImageFile = createFile(takeImageFile, "IMG_", ".jpg");     photoUri = FileProvider.getUriForFile(this, getPackageName() + ".fileprovider", takeImageFile);     intent.putExtra(MediaStore.EXTRA_OUTPUT, photoUri);     activity.startActivityForResult(intent, requestCode); }

4. 拓展:获取图片方式

4.1 通过绝对路径获取图片

image.setImageBitmap(BitmapFactory.decodeFile(takeImageFile.getAbsolutePath()));

4.2 通过Uri获取资源流

Bitmap bitmap = null; try {     bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(photoUri)); } catch (FileNotFoundException e) {     e.printStackTrace(); } image2.setImageBitmap(bitmap);

4.3 通过cursor获取图片

这是错误示范,仅供参考:

Cursor cursor = getContentResolver().query(photoUri, new String[]{MediaStore.Images.Media.DATA}, null, null, null); if (cursor != null) {     if (cursor.getColumnCount() > 0) {         if (cursor.moveToFirst()) {             int index = cursor.getColumnIndex(MediaStore.Images.Media.DATA);             if (index > -1) {                 path = cursor.getString(index);             } else {                 LogUtils.e(TAG, "cursor index -1");             }         }     }     cursor.close(); }

通过Uri和selection来获取真实路径的方法在Android系统中是可行的,但由于我们的Uri不是目标文件的绝对路径,查询不到任何信息。

尝试自己拼接Uri来查询也没有成功:

Uri uri = new Uri.Builder()     .scheme("content")     .authority(BuildConfig.APPLICATION_ID+".fileprovider")     .path("/DCIM/camerademo/IMG_20200527_093044.jpg")     .build();

打印出的Uri为:

uri ===content://com.yoshin.company.blogdemo.fileprovider/DCIM/camerademo/IMG_20200527_093044.jpg

替换photoUri去查询,仍然查询不到任何信息。

严格模式解决办法

项目中的解决办法是在onCreate时添加如下代码:

// android 7.0系统解决拍照的问题 StrictMode.VmPolicy.Builder builder = new StrictMode.VmPolicy.Builder(); StrictMode.setVmPolicy(builder.build()); builder.detectFileUriExposure();

虽然官方不推荐这种方法,但存在即合理。

摄像通过这段代码,摄像机也可以这么搞

private void takenVedio() {     // 表示跳转至相机的录视频界面     Intent intent = new Intent(MediaStore.ACTION_VIDEO_CAPTURE);     // MediaStore.EXTRA_VIDEO_QUALITY 表示录制视频的质量,从 0-1,越大表示质量越好,同时视频也越大     intent.putExtra(MediaStore.EXTRA_VIDEO_QUALITY, 0);     intent.putExtra(MediaStore.EXTRA_DURATION_LIMIT, 30);     // 调用前置摄像头     intent.putExtra("android.intent.extras.CAMERA_FACING", 1);     intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);     if (intent.resolveActivity(getPackageManager()) != null) {         File file;         if (existSDCard()) {             file = new File(Environment.getExternalStorageDirectory(), "/DCIM/audio/");         } else {             file = Environment.getDataDirectory();         }         Log.i(TAG, "file ===" + file);         file = createFile(file, "VIDEO_", ".mp4");         Log.i(TAG, "file ===" + file);         if (file != null) {             Uri uri = Uri.parse("file://" + file.getAbsolutePath());             Log.i(TAG, "uri ===" + uri);             intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);         }     }     startActivityForResult(intent, REQUEST_CODE_RECORD_VIDEO); }

总结

通过使用FileProvider,可以有效解决Android 7.0及以上版本中相机和摄像头在严格模式下的问题。希望本文对你有所帮助,欢迎大神留言指正。

推荐github开源项目 SelectImgAsWechath:https://www.php.cn/link/cd9015a609618aa8dcd0a7ff7941c475

参考地址:Android FileProvider详细解析和踩坑指南

发布者:全栈程序员栈长,转载请注明出处:https://www.php.cn/link/d0241607c7a6d2011c767cfb35b8248d

原文链接:https://www.php.cn/link/c8377ad2a50fb65de28b11cfc628d75c

© 版权声明
THE END
喜欢就支持一下吧
点赞6 分享