1.0 APP在程序内部进行操作,当然是安全的且合理的,但不同应用程序之间如何进行数据访问?比如这个例子,需要自己的APP利用到手机上的拨打电话权限。
2.0 在Android6.0系统之后,Android开发团队引用了运行时权限这个功能,终于可以出“店大欺客”这口恶气,这样即使我们不赋予APP某项权限,仍可以安装并使用该软件。
这个网站有关于Android系统中完整的权限列表:
链接:Android API:Manifest.permission权限清单
(不好意思,浏览器已中文翻译,虽然这翻译是磕碜了点)
3.0 话不多说,通过一个简单的小例子,体验一把到底什么是Android运行时权限
新建项目,RuntimePermissionTest,目录结构如下:
4.0 在activity_main.xml中放一个button按钮控件。
<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity">
<Button
android:id="@+id/make_call"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="拨打电话"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintVertical_bias="0.05" />
</android.support.constraint.ConstraintLayout>
5.0 接着在AndroidManifest.xml中增加一行<uses-permission android:name="android.permission.CALL_PHONE"/>
:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.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 MainActivity.java,说明都在代码里,总之步骤分为三步走。
package com.example.runtimepermissiontest;
import android.Manifest;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.net.Uri;
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 {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Button makeCall = (Button) findViewById(R.id.make_call);
makeCall.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//第一步,判断用户是否给应用授权了
//借助ContextCompat.checkSelfPermission()方法
//该方法接收两个参数一个是上下文,没什么好说的。
//第二个是具体的权限名
//然后把方法的返回值和PackageManager.PERMISSION_GRANTED作比较,相等说明用户已经授权
if (ContextCompat.checkSelfPermission(MainActivity.this,
Manifest.permission.CALL_PHONE) != PackageManager.PERMISSION_GRANTED) {
//第二步,如果不等于,则需要调用 ActivityCompat.requestPermissions()方法来向用户申请权限
//该方法接收三个参数,第一个要求是Activity的实例
//第二个是一个String数组,把需要申请的权限名放入数组即可
//第三个是请求码,是唯一值即可,这里用的“1”
ActivityCompat.requestPermissions(MainActivity.this,
new String[]{Manifest.permission.CALL_PHONE}, 1);
} else {
call();
}
}
});
}
private void call() {
try {
Intent intent = new Intent(Intent.ACTION_CALL);
intent.setData(Uri.parse("tel:10010"));
startActivity(intent);
} catch (SecurityException e) {
e.printStackTrace();
}
}
//第三步,调用完requestPermissions()方法后,系统会弹出一个权限申请的对话框
// 无论结果如何,最终都会回调onRequestPermissionsResult()方法
//授权的结果,会封装在grantResults中。
// 判断一下,如果同意了授权就打电话,没有就凉凉了……
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions,
int[] grantResults) {
switch (requestCode) {
case 1:
if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
call();
} else {
Toast.makeText(this, "抱歉,没有该权限!", Toast.LENGTH_SHORT).show();
}
}
}
}
7.0 跑一下,最后的效果如下:
点击“拨打电话”,提示如下:
选择,拒绝:
选择允许:
END