拍照(调用相机)
调用系统相机要注意的几个问题:
-
Android 6.0+ 权限问题
- Android版本6.0以上(含6.0)需要申请权限
这里就贴下代码:
首先加入权限
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> <uses-permission android:name="android.permission.CAMERA"/>
申请权限(这里除了相机权限,调用相册还需要文件读写权限)
//检查权限 int ret = ActivityCompat.checkSelfPermission(ActivateActivity.this, Manifest.permission .CAMERA); //申请权限 if (ret != PackageManager.PERMISSION_GRANTED) { ActivityCompat.requestPermissions(this,new String[] {Manifest.permission.android.Manifest.permission.CAMERA)}, RESULT_CODE_CAMERA); return; } } //回调 @Override public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) { switch (requestCode) { case RESULT_CODE_CAMERA: //TODO break;
- Android版本6.0以上(含6.0)需要申请权限
-
Camera API使用问题
- 在API21中Google发布了Camera2类来取代Camera类,所以当API版本在21以上可使用Camera2,API版本在21以下必须使用Camera;
-
Android7.0调用相机问题
-
android升级到7.0后对权限又做了一个更新即应用间传递文件时不允许出现以file://的形式 调用隐式APP,需要用共享文件的形式:content:// URI,如果给出一个file://格式的URI的话,应用会抛出FileUriExposedException。这是由于谷歌认为目标app可能不具有文件权限,会造成潜在的问题,这里有两种解决方式
-
使用共享文件的形式(FileProvider方式)
这是谷歌官方推荐的解决方案。即使用FileProvider来生成一个content://格式的URI。具体实现方式如下:-
在manifest文件中声明
//android:name值是固定的 //android:authorities随便写但是必须得保证唯一性,这边用的是包名 //android:grantUriPermission和android:exported 是固定值 <manifest> ... <application> ... <provider android:name="android.support.v4.content.FileProvider" android:authorities="com.test.fileprovider" android:exported="false" android:grantUriPermissions="true"> <meta-data android:name="android.support.FILE_PROVIDER_PATHS" android:resource="@xml/file_paths" /> </provider> ... </application> </manifest>
-
在res文件夹下新建一个xml文件夹,在xml文件夹下新建file_paths.xml文件
//paths标签下的子元素一共有以下几种: //files-path 对应 Context.getFilesDir() //cache-path 对应 Context.getCacheDir() //external-path 对应 Environment.getExternalStorageDirectory() //external-files-path 对应 Context.getExternalFilesDir() //external-cache-path 对应 Context.getExternalCacheDir() <paths xmlns:android="http://schemas.android.com/apk/res/android"> //name表示生成URI时的别名,path是指相对路径。 <files-path name="my_images" path="images/"/> ... </paths>
-
使用
String filePath = Environment.getExternalStorageDirectory() + "/images/"+System.currentTimeMillis()+".jpg"; File outputFile = new File(filePath); if (!outputFile.getParentFile().exists()) { outputFile.getParentFile().mkdir(); } //创建content URI Uri imgUri = FileProvider.getUriForFile(this,"com.test.fileprovider", outputFile); Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); intent.putExtra(MediaStore.EXTRA_OUTPUT, imgUri); intent.addFlags(FLAG_GRANT_READ_URI_PERMISSION); intent.addFlags(FLAG_GRANT_WRITE_URI_PERMISSION); startActivityForResult(intent, REQUEST_TAKE_PICTURE); //在回调onActivityForResult中获取图片URI @Override public void onActivityResult(int requestCode, int resultCode, Intent data) { super.onActivityResult(requestCode, resultCode, data); switch (requestCode){ case REQUEST_TAKE_PICTURE: Uri imgUri = FileProvider.getUriForFile(getActivity(), "com.test.fileprovider", mCameraFile); break; }
-
-
开启严苛模式
Android 2.3提供一个称为严苛模式(StrictMode)的调试特性,
它将报告与线程及虚拟机相关的策略违例。一旦检测到策略违例(policy violation),你将获得警告,其包含了一个栈trace显示你的应用在何处发生违例。你可以强制用警告代替崩溃(crash),也可以仅将警告计入日志,让你的应用继续执行;严格模式的开启可以放在Application或者Activity以及其他组件的onCreate方法。为了更好地分析应用中的问题,建议放在Application的onCreate方法中。
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { StrictMode.VmPolicy.Builder builder = new StrictMode.VmPolicy.Builder().detectFileUriExposure(); StrictMode.setVmPolicy(builder.build()); }
-
-
网友评论