GitHub项目地址:

https://github.com/Skymqq/RuntimePermissionTest.git

运行时权限是Android6.0系统引入的新特性,那么为何要引入这种运行时权限的机制呢?这是为了保障用户的安全和隐私,如果系统声明了某些危险权限,而这个时候用户却一点都不知晓,这就显得很不人性化,如果在程序安装界面,弹出一个对话框来通知用户,是否允许当前程序拥有一些权限。这就是运行时权限最常用的场景。

下面我们新建一个RuntimePermissionTest项目来实现调用系统电话薄的功能,期间通过运行时权限功能来让我们抉择是否允许声明打电话的权限。

activity_main.xml代码:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="vertical"><Buttonandroid:id="@+id/btn_call"android:layout_width="match_parent"android:layout_height="wrap_content"android:text="Make Call"android:textAllCaps="false"android:textSize="18sp"android:textStyle="bold" />
</LinearLayout>

MainActivity.java代码:

package com.example.administrator.runtimepermissiontest;import android.content.Intent;
import android.net.Uri;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;public class MainActivity extends AppCompatActivity {private Button btn_call;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);initView();//初始化UI控件}private void initView() {btn_call = (Button) findViewById(R.id.btn_call);}@Overrideprotected void onResume() {super.onResume();try {btn_call.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {Intent intent = new Intent(Intent.ACTION_CALL);//ACTION_CALL是系统内置的打电话的动作intent.setData(Uri.parse("tel:10086"));//tel是协议,10086是号码startActivity(intent);}});} catch (Exception e) {e.printStackTrace();}}
}

可以看到,在按钮的点击事件中,我们构建了一个隐式Intent,Intent的action指定为Intent.ACTION_CALL,这是一个系统内置的打电话的动作,然后在data部分指定了协议是tel,号码是10086.另外为了防止程序崩溃,我们将所有操作都放在了异常捕获代码块当中。

最后在AndroidManifest中声明权限:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"package="com.example.administrator.runtimepermissiontest"><uses-permission android:name="android.permission.CALL_PHONE" /><applicationandroid:allowBackup="true"android:icon="@mipmap/ic_launcher"android:label="@string/app_name"android:roundIcon="@mipmap/ic_launcher_round"android:supportsRtl="true"android:theme="@style/AppTheme"><activity android:name=".MainActivity"><intent-filter><action android:name="android.intent.action.MAIN" /><category android:name="android.intent.category.LAUNCHER" /></intent-filter></activity></application></manifest>

运行程序后,如果你的设备是6.0以下的版本,那么在你点击Make Call按钮后,程序会直接打开电话薄,如果你的设备室6.0以上版本,那么你的程序会打不开电话薄,同时会报错“Permission Denial”,这是由于权限被禁止所导致的,因为6.0以上系统在使用危险权限时,都必须进行运行时权限处理。

那么下面我们就来尝试着修复这个问题,修改MainActivity.java代码,如下所示:

package com.example.administrator.runtimepermissiontest;import android.Manifest;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.net.Uri;
import android.support.annotation.NonNull;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.ContextCompat;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;public class MainActivity extends AppCompatActivity {private Button btn_call;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);initView();//初始化UI控件}private void initView() {btn_call = (Button) findViewById(R.id.btn_call);}@Overrideprotected void onResume() {super.onResume();btn_call.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.CALL_PHONE) != PackageManager.PERMISSION_GRANTED) {ActivityCompat.requestPermissions(MainActivity.this, new String[]{Manifest.permission.CALL_PHONE}, 1);} else {call();//打电话}}});}private void call() {try {Intent intent = new Intent(Intent.ACTION_CALL);//ACTION_CALL是系统内置的打电话的动作intent.setData(Uri.parse("tel:10086"));//tel是协议,10086是号码startActivity(intent);} catch (Exception e) {e.printStackTrace();}}@Overridepublic void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {switch (requestCode) {case 1:if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {call();//打电话} else {Toast.makeText(this, "You denied the permission", Toast.LENGTH_SHORT).show();}break;default:}}
}

上面的代码将运行时权限的完整流程都覆盖了,下面我们来具体解析一下。说白了,运行时权限的核心就是在程序运行的过程中由用户授权我们去执行某些危险操作,程序是不可以擅自做主去执行这些危险操作的。因此,第一步就是要先判断用户是不是已经给过我们授权了,借助的是ContextCompat.checkSelfPermission()。checkSelfPermission方法接收两个参数,第一个参数是Context,这个没什么好说的,第二个参数是具体的权限名,比如打电话的权限名就是 Manifest.permission.CALL_PHONE,然后我们使用方法的返回值和PackManager.PERMISSION_GRANTED做比较,如果值相等就说明用户已经授权,如果不相等就表示用户没有授权。

如果已经授权就很简单了,直接去执行拨打电话的逻辑操作就可以了,这里我们把拨打电话的逻辑封装到了call()方法当中。如果没有授权的话,则需要调用ActivityCompat.requestPermission()方法来向用户申请授权,requestPermission()方法接收3个参数,第一个参数要求Activity的实例,第二个参数是一个String数组,我们把要申请的权限名放在数组中即可,第三个参数是请求码,只要是唯一值就可以了,这里就传入了个1.

调用玩了requestPermission()方法之后,系统会弹出一个权限申请的对话框,然后用户可以选择同意或拒绝我们的权限申请,不论是那种结果,最终都会回调到onRequestPermissionResult()方法中,而授权的结果则会封装在grantResults参数当中。这里我们只需要判断一下最后的授权结果,如果用户同意的话就调用call()方法来拨打电话,如果用户拒绝的话我们只能放弃操作,并且弹出一条拒绝权限的失败提示。

运行程序:

点击Make Call按钮之后会弹出一个对话框,这个对话框是系统自带的:

点击允许后直接打开系统内置电话薄,并且拨打了10086:

如果你一开始点击的拒绝,就会弹出Toast提示。如果你在允许授权之后,又希望取消授权拨打电话的权限,可以主动去设置里面将拨打电话的权限给取消就行了。

(调用系统电话薄)运行时权限的基本使用相关推荐

  1. Android运行时权限,设置帮助类BaseActivity;电话权限,短信权限,

                                      Android运行时权限设置帮助类BaseActivity  一.电话权限 package com.glandroid.smssen ...

  2. Android权限系统(三):运行时权限检查和申请,PermissionController

    请求应用权限的最佳实践   Google提供的请求应用权限的说明如下:请求应用权限   官方提供的模板使用了三个条件分支来请求应用权限:   1.checkSelfPermission用来检查应用是否 ...

  3. Android 11.0 系统默认授予app安装权限(去掉app首次运行时权限授权弹窗)

    目录 1.概述 2.系统默认授予app安装权限去掉app首次运行时权限授权弹窗功能分析

  4. 运行时权限+读取系统联系人

    GitHub项目地址: https://github.com/Skymqq/ContactsTest.git 首先在模拟器中创建2个联系人,如下所示: 我们看到列表中中已经存在2位联系人了. 新建一个 ...

  5. Android8.0运行时权限策略变化和适配方案

    版权声明:转载必须注明本文转自严振杰的博客:http://blog.yanzhenjie.com Android8.0也就是Android O即将要发布了,有很多新特性,目前我们可以通过Android ...

  6. Android 6.0 运行时权限处理完全解析

    一.概述 随着Android 6.0发布以及普及,我们开发者所要应对的主要就是新版本SDK带来的一些变化,首先关注的就是权限机制的变化.对于6.0的几个主要的变化,查看查看官网的这篇文章http:// ...

  7. Android 8.0 运行时权限策略变化和适配方案

    Android8.0也就是Android O即将要发布了,有很多新特性,目前我们可以通过AndroidStudio3.0 Canary版本下载Android O最新的系统映像的Developer Pr ...

  8. Android 8.0学习(18)--- Android8.0运行时权限策略变化和适配方案

    Android8.0运行时权限策略变化和适配方案    在 Android O 之前,如果应用在运行时请求权限并且被授予该权限,系统会错误地将属于同一权限组并且在清单中注册的其他权限也一起授予应用.对 ...

  9. Android 6.0运行时权限管理

    运行时权限管理 定义: 之前我们的App需要权限,只需在manifest中申明,用户安装后,一切申明的权限都可使用. 但是Android 6.0以后Android M把权限管理做了加强处理,除了需要在 ...

最新文章

  1. 详细记录python的range()函数用法
  2. c# imager让图片有圆角unity_Qt编写自定义控件24-图片轮播控件
  3. 《走遍中国》珍藏版(一)
  4. 过了一个有意义的愚人节
  5. 判断是否是完全二叉树_二叉树:我对称么?
  6. python 求当前小时前n个小时
  7. Mac安装svn客户端
  8. Gallery 是国外一个免费开源的图库相册软件
  9. oracle 除数不能为0
  10. 数据库迁移测试——功能测试
  11. 解决win8.1中文版无法添加域的问题
  12. Cousera Sequence,Time Series and PredictionWK1
  13. matlab中position用法_Matlab中的zeta函数用法
  14. 中国大学MOOC公开信息收集
  15. 我在 B 站,被喷了。。。
  16. 人物画像————圆球转动效果
  17. 推荐系统三十六式(刑无刀)学习笔记(一)
  18. STM32 堆栈大小的设置及分析
  19. IAP 促销优惠(promotional offer)后端接入指南
  20. 无线双向网助贵州广电打造融合网络

热门文章

  1. 【PC工具】更新win10关闭更新工具及注意事项
  2. 【模块】ESP32CAM arduino程序下载方法及注意事项避坑笔记
  3. 通过MATLAB提取图像的深度信息
  4. Zepto源码分析-架构
  5. 用系统滚动条实现NumericUpDown的原理
  6. 【全栈项目上线(vue+node+mongodb)】06.nodejs服务上线(生产环境前后分离的vue项目中怎么解决跨域问题)...
  7. TSC的分歧导致Node.js分支
  8. CUBA在查询语句如何添加参数
  9. 配置管理篇(0)_SaltStack 安装
  10. 性能计数器取网卡流量