本文结合代码和图示来展示安卓运行时权限的申请流程:
安卓中,运行时权限的核心就是需要在程序运行的过程当中,去判断用户是否对我们授权以允许相关操作。
开始之前,先贴上动图看看效果:
其实,在代码层面,安卓已经为我们封装好了实现的方法。
我们只需要调用这些方法就可以了。
所以最重要的是这一流程的逻辑。
既然是要得到授权。
第一步,就是要判断用户是否已经给过授权了。
这一步可以借助于Contextcompat.checkSelfPermission()方法。
此方法接收两个参数:
1. context对象
2. 具体想要获得授权的权限名称,如打电话的权限就是Manifest.permission.CALL_PHONE
然后使用该方法的返回值和PackageManager.PERMISSION_GRANTED作比较。
如果相等,则说明用户已经授权;若不等,就表示用户没有授权。
第二步,如果用户没有授权,就需要申请用户授权:
调用ActivityCompat.requestPermissions()方法来向用户申请授权。
其中,ActivityCompat.requestPermissions()方法接收三个参数:
1. Activity的实例
2. String类型的数组,我们把要申请的权限的权限名放入其中即可
3. 请求码:只要是唯一值即可:比如传入1
第三步,在调用完ActivityCompat.requestPermissions()方法之后,系统会弹出对话框来提示用户是否对该权限授权。用户既可以选择同意,也可以选择拒绝。
但是无论用户怎么选择,
最终的结果都会回调到onRequestPermissionResult()方法中。
而授权的结果则会封装在grantResults参数中。
第四步,判断最后的授权结果,如果用户同意则执行逻辑。
如果用户不同意,则只能放弃操作,并弹出授权失败的提示。
当然,文字表述不利于清楚地表现其中的逻辑。
这里贴出对应的流程图,作为补充:
下面再贴出在具体代码中的应用:
public class MainActivity extends AppCompatActivity {
//定义按钮
Button makeCall;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
makeCall = (Button) findViewById(R.id.bn_make_call);
//为按钮设置事件监听器
makeCall.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
//先检查用户有没有同意权限申请
//用户未曾授权,则请求授权
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(); //call()方法中封装了打电话的逻辑
}
}
});
}
//打电话方法的逻辑
public void call() {
try {
//发送一个隐式的Intent
Intent intent = new Intent(Intent.ACTION_CALL);
//设置协议为 tel
intent.setData(Uri.parse("tel:1234567890"));
startActivity(intent);
} catch (SecurityException e) {
e.printStackTrace();
}
}
@Override
public void onRequestPermissionsResult(int requestCode,
@NonNull String[] permissions,
@NonNull int[] grantResults)
{
super.onRequestPermissionsResult(requestCode,
permissions, grantResults);
//这里的requestCode与我们之前
//在requestPermission()方法中传入的请求码对应
switch (requestCode){
case 1:
//授权结果被封装在grantResults中,注意其为int类型的数组
//用户授权,则执行相应操作
if (grantResults.length >0 &&
grantResults[0]==PackageManager.PERMISSION_GRANTED){
call();
}else{
//用户拒绝授权,则只能放弃,并弹出提示
Toast.makeText(this,
"You denied the permission",
Toast.LENGTH_SHORT).show();
}
break;
default:
}
}
}
诸君共勉:)