Android开发之 permission动态权限获取

本文详细介绍了在Android 6.0及其以上版本中,如何处理运行时权限请求。针对读取联系人、读取外部存储和写入外部存储等危险权限,需要在运行时动态请求用户授权。通过`ActivityCompat`和`ContextCompat`进行权限检查,当缺少权限时,使用`requestPermissions()`发起请求,并在`onRequestPermissionsResult()`中处理结果。如果用户拒绝授权,可以引导用户进入应用设置界面手动开启权限。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

说明

  • Android 6.0 (API 23) 之前应用的权限在安装时全部授予,运行时应用不再需要询问用户。在 Android 6.0 或更高版本对权限进行了分类,对某些涉及到用户隐私的权限可在运行时根据用户的需要动态授予。这样就不需要在安装时被强迫同意某些权限。
  • 设备系统是 Android 6.0 (API 23) 或更高版本,并且应用的 targetSdkVersion 是 23 或更高版本,则针对在 AndroidManifest.xml 中声明的危险权限,在运行时还需要动态请求用户授权
  • 动态权限请求相关操作的API封装在在android.support.v4包中,发起请求权限的Activity需要直接或间接继承android.support.v4.app.FragmentActivity。
  • 也可以在直接或间接继承 android.support.v4.app.Fragment 的 Fragment 中发起权限请求。
	package com.xiets.demoapp;
	 
	import android.Manifest;
	import android.content.DialogInterface;
	import android.content.Intent;
	import android.content.pm.PackageManager;
	import android.net.Uri;
	import android.os.Bundle;
	import android.provider.Settings;
	import android.support.annotation.NonNull;
	import android.support.v4.app.ActivityCompat;
	import android.support.v4.content.ContextCompat;
	import android.support.v7.app.AlertDialog;
	import android.support.v7.app.AppCompatActivity;
	import android.view.View;
	import android.widget.Toast;
	 
	/**
	 * 一键备份通讯录
	 *
	 * @author xietansheng
	 */
	public class MainActivity extends AppCompatActivity {
	 
	    private static final int MY_PERMISSION_REQUEST_CODE = 10000;
	 
	    @Override
	    protected void onCreate(Bundle savedInstanceState) {
	        super.onCreate(savedInstanceState);
	        setContentView(R.layout.activity_main);
	    }
	 
	    /**
	     * 点击按钮,将通讯录备份保存到外部存储器备。
	     *
	     * 需要3个权限(都是危险权限):
	     *      1. 读取通讯录权限;
	     *      2. 读取外部存储器权限;
	     *      3. 写入外部存储器权限.
	     */
	    public void click(View view) {
	        /**
	         * 第 1 步: 检查是否有相应的权限,根据自己需求,进行添加相应的权限
	         */
	        boolean isAllGranted = checkPermissionAllGranted(
	                new String[] {
	                        Manifest.permission.READ_CONTACTS,
	                        Manifest.permission.READ_EXTERNAL_STORAGE,
	                        Manifest.permission.WRITE_EXTERNAL_STORAGE
	                }
	        );
	        // 如果这3个权限全都拥有, 则直接执行备份代码
	        if (isAllGranted) {
	            doBackup();
	            return;
	        }
	 
	        /**
	         * 第 2 步: 请求权限
	         */
	        // 一次请求多个权限, 如果其他有权限是已经授予的将会自动忽略掉
	        ActivityCompat.requestPermissions(
	                this,
	                new String[] {
	                        Manifest.permission.READ_CONTACTS,
	                        Manifest.permission.READ_EXTERNAL_STORAGE,
	                        Manifest.permission.WRITE_EXTERNAL_STORAGE
	                },
	                MY_PERMISSION_REQUEST_CODE
	        );
	    }
	 
	    /**
	     * 检查是否拥有指定的所有权限
	     */
	    private boolean checkPermissionAllGranted(String[] permissions) {
	        for (String permission : permissions) {
	            if (ContextCompat.checkSelfPermission(this, permission) != PackageManager.PERMISSION_GRANTED) {
	                // 只要有一个权限没有被授予, 则直接返回 false
	                return false;
	            }
	        }
	        return true;
	    }
	 
	    /**
	     * 第 3 步: 申请权限结果返回处理
	     */
	    @Override
	    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
	        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
	 
	        if (requestCode == MY_PERMISSION_REQUEST_CODE) {
	            boolean isAllGranted = true;
	 
	            // 判断是否所有的权限都已经授予了
	            for (int grant : grantResults) {
	                if (grant != PackageManager.PERMISSION_GRANTED) {
	                    isAllGranted = false;
	                    break;
	                }
	            }
	 
	            if (isAllGranted) {
	                // 如果所有的权限都授予了, 则执行备份代码
	                doBackup();
	 
	            } else {
	                // 弹出对话框告诉用户需要权限的原因, 并引导用户去应用权限管理中手动打开权限按钮
	                openAppDetails();
	            }
	        }
	    }
	 
	    /**
	     * 第 4 步: 备份通讯录操作
	     */
	    private void doBackup() {
	        // 本文主旨是讲解如果动态申请权限, 具体备份代码不再展示, 就假装备份一下
	        Toast.makeText(this, "正在备份通讯录...", Toast.LENGTH_SHORT).show();
	    }
	 
	    /**
	     * 打开 APP 的详情设置
	     */
	    private void openAppDetails() {
	        AlertDialog.Builder builder = new AlertDialog.Builder(this);
	        builder.setMessage("备份通讯录需要访问 “通讯录” 和 “外部存储器”,请到 “应用信息 -> 权限” 中授予!");
	        builder.setPositiveButton("去手动授权", new DialogInterface.OnClickListener() {
	            @Override
	            public void onClick(DialogInterface dialog, int which) {
	                Intent intent = new Intent();
	                intent.setAction(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
	                intent.addCategory(Intent.CATEGORY_DEFAULT);
	                intent.setData(Uri.parse("package:" + getPackageName()));
	                intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
	                intent.addFlags(Intent.FLAG_ACTIVITY_NO_HISTORY);
	                intent.addFlags(Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS);
	                startActivity(intent);
	            }
	        });
	        builder.setNegativeButton("取消", null);
	        builder.show();
	    }
	 
	}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值