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

本文深入探讨了Android 6.0引入的运行时权限机制,旨在保护用户隐私和安全。通过一个具体示例,详细解释了如何在应用程序中实现运行时权限检查,包括调用系统电话簿功能的代码实现。

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">

    <Button
        android: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;

    @Override
    protected 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);
    }

    @Override
    protected void onResume() {
        super.onResume();
        try {
            btn_call.setOnClickListener(new View.OnClickListener() {
                @Override
                public 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" />

    <application
        android: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;

    @Override
    protected 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);
    }

    @Override
    protected void onResume() {
        super.onResume();
        btn_call.setOnClickListener(new View.OnClickListener() {
            @Override
            public 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();
        }
    }


    @Override
    public 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提示。如果你在允许授权之后,又希望取消授权拨打电话的权限,可以主动去设置里面将拨打电话的权限给取消就行了。

 

 

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值