大家好,又见面了,我是你们的朋友全栈君。
今天我们来讨论一下在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配置
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