随着Android版本越来越高, Android对用户隐私的保护力度也越来越大。从Android6.0引入的动态权限控制到Android7.0的私有目录被限制访问,“StrictMode API 政策”。这些更改在为用户带来更加安全的操作系统同时也为我们开发者带来了新的任务。

系统6.0,声明权限,开启权限就ok了。

1.先判断用户是否授予某权

private void checkPermisson() {
    //第二个参数是需要申请的权限
    if (ContextCompat.checkSelfPermission(this, Manifest.permission.CAMERA)
            != PackageManager.PERMISSION_GRANTED) {
        //权限还没有授予,需要在这里写申请权限的代码
        ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.CAMERA, Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1);
    } else {
        //权限已经被授予,在这里直接写要执行的相应方法即可
    }
}
2.回调onRequestPermissionsResult()
//判断用户是否授予该权限
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
    super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    doNext(requestCode, grantResults);
}

private void doNext(int requestCode, int[] grantResults) {
    if (requestCode == 1) {
        if (grantResults[0] == PackageManager.PERMISSION_GRANTED && grantResults.length == 2) {
            // Permission Granted
            showSelectPicDialog();
            Log.e("monster", "open");
        } else {
            // Permission Denied
            Tips.instance.tipShort("请在应用管理中打开“相机”和“sd卡”访问权限!");
            Log.e("monster", "close");
        }
    }
}
权限开启后,拍照就没问题了。可是一到android7.0的手机测试,拍照崩溃,图库选择图片进行裁剪也崩溃。
1.manifests.xml 注册provider
exported:要求必须为false,为true则会报安全异常。grantUriPermissions:true,表示授予 URI 临时访问权限


<provider
    android:name="android.support.v4.content.FileProvider"
    android:authorities="com.bj.huitianxia.provider"    //包名.provider(可自己取)
    android:exported="false"
    android:grantUriPermissions="true">
    <meta-data
        android:name="android.support.FILE_PROVIDER_PATHS"   
        android:resource="@xml/file_paths" />
</provider>
2.指定共享的目录: 在res下创建xml文件夹,xml下创建file_paths
代表的根目录: Environment.getExternalStorageDirectory()
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <!--external-path 就是用来指定URI共享的-->
    <!--name属性的值可以随便写-->
    <!--path属性的值表示共享的具体位置,设置空就表示将整个SD卡进行共享-->
    <!--用该路径保存的好处是,文件会随着应用的卸载而删除,file/header/为自定义路径-->
    <external-path path="" name="camera_photos" />
</paths>

3.使用FilePeovider

//拍照
private void getImageFromCamera() {
    if (SdcardUtils.checkSDCard()) {
        Uri imageUri = null;
        Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        File file = FileUtils.createFile(Constant.FILE_ROOT_NAME,
                picName,
                Constant.PICTURE_TYPE);
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { //7.0
            //兼容android7.0 使用共享文件的形式
            imageUri = FileProvider.getUriForFile(this, "com.bj.huitianxia.provider", file);//通过FileProvider创建一个content类型的Uri
            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); //添加这一句表示对目标应用临时授权该Uri所代表的文件
        } else {
            imageUri = Uri.fromFile(file);
        }
        //将拍取的照片保存到指定uri
        intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
        startActivityForResult(intent, REQUEST_CODE_CAPTURE_CAMEIA);
    }
}
@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);
    switch (requestCode) {
        case REQUEST_CODE_CAPTURE_CAMEIA: //拍照
            startPhotoZoom(Uri.fromFile(FileUtils.createFile(Constant.FILE_ROOT_NAME,
                    picName, Constant.PICTURE_TYPE)));
            break;
        case REQUEST_CODE_CAPTURE_ALBUM: //图库
            if (null != data) {
                startPhotoZoom(data.getData());
            }
            break;
        case REQUEST_CODE_CAPTURE_CROP://裁剪
            if (data != null) {
                path = SdcardUtils.getSDCardPathWithFileSeparators() + Constant.FILE_ROOT_NAME
                        + File.separator + picName + Constant.PICTURE_TYPE;
                file = new File(path);
                if (file.exists()) {
                    postHeadFile();
                } else {
                    Tips.instance.tipShort("图片保存失败,请重新选择");
                }
            }
            break;
    }
}



//调用系统裁剪
    private void startPhotoZoom(Uri uri) {
        Uri outPutCutUri = null;
        Uri imageUri = null;

        Intent intent = new Intent("com.android.camera.action.CROP");
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N && type == 1) { //7.0
            File file = FileUtils.createFile(Constant.FILE_ROOT_NAME,
                    picName,
                    Constant.PICTURE_TYPE);
            if (!file.exists()){
                Tips.instance.tipShort("图片不存在,请重新选择");
                return;
            }
            outPutCutUri = Uri.fromFile(FileUtils.createFile(Constant.FILE_ROOT_NAME,
                    picName,
                    Constant.PICTURE_TYPE));
//            outPutCutUri = FileProvider.getUriForFile(this, "com.bj.huitianxia.provider", file);
            imageUri = FileProvider.getUriForFile(this, "com.bj.huitianxia.provider", file);//通过FileProvider创建一个content类型的Uri
            //添加这一句表示对目标应用临时授权该Uri所代表的文件
            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
        } else {
            outPutCutUri = Uri.fromFile(FileUtils.createFile(Constant.FILE_ROOT_NAME,
                    picName,
                    Constant.PICTURE_TYPE));
            imageUri = uri;
        }
        intent.setDataAndType(imageUri, "image/*")
                .putExtra("crop", "true")
                .putExtra("outputX", 80)
                .putExtra("outputY", 80)
                .putExtra("return-data", false)
                .putExtra("outputFormat", Bitmap.CompressFormat.JPEG)
                .putExtra(MediaStore.EXTRA_OUTPUT, outPutCutUri)
                .putExtra("aspectX", 1)
                .putExtra("aspectY", 1);
        startActivityForResult(intent, REQUEST_CODE_CAPTURE_CROP);
    }


另外,推荐大家使用开源工具库TakePhoto,
TakePhoto是一款在Android设备上获取照片(拍照或从相册、文件中选择)、裁剪图片、压缩图片的开源工具库。