普通权限没有列出,因为除了危险权限就是普通权限。
具体的危险权限组和权限如下:
-
CALENDAR
-
READ_CALENDAR
-
WRITE_CALENDAR
-
CAMERA
-
CAMERA
-
CONTACTS
-
READ_CONTACTS
-
WRITE_CONTACTS
-
GET_ACCOUNTS
-
LOCATION
-
ACCESS_FINE_LOCATION
-
ACCESS_COARSE_LOCATION
-
MICROPHONE
-
RECORD_AUDIO
-
PHONE
-
READ_PHONE_STATE
-
CALL_PHONE
-
READ_CALL_LOG
-
WRITE_CALL_LOG
-
ADD_VOICEMAIL
-
USE_SIP
-
PROCESS_OUTGOING_CALLS
-
SENSORS
-
BODY_SENSORS
-
SMS
-
SEND_SMS
-
RECEIVE_SMS
-
READ_SMS
-
RECEIVE_WAP_PUSH
-
RECEIVE_MMS
-
STORAGE
-
READ_EXTERNAL_STORAGE
-
WRITE_EXTERNAL_STORAGE
另外注意:表格中每一个危险权限都属于一个权限组,我们在进行运行时权限处理时使用的是权限名,但是用户一旦同一授权了,那么该权限所对应的权限组中所有的其他权限也会同时被授权。
在运行时权限:
说白了用过智能手机的都知道,也就是在刚刚下载的App打开时,可能是照相,那么系统将会询问你是否授权摄像头权限,打开相册时是否允许访问相册,所以在这里运行时的权限是很好理解的,不再解释那么多。
下面来新建一个项目RuntimePermissionTest项目。
这里我们使用CALL_PHONE这个权限来作为示例吧。
CALL_PHONE这个权限是编写拨打电话功能的时候需要声明的,因为拨打电话会涉及用户手机的资费问题,因而被列为了危险权限。
先修改activity_main.xml布局文件,如下所示:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
>
<Button
android:id="@+id/make_call"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="make Call"
/>
</LinearLayout>
当点击按钮时就去触发拨打电话的逻辑。
修改MainActivity
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()方法。
//checkSelfPermission()方法接收两个参数,第一个参数是Context,第二个参数是权限名。
//然后使用方法的返回值和PackageManager.PERMISSION_GRANTED作比较,相等就说明用户已经授权。
if(ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.CALL_PHONE)!= PackageManager.PERMISSION_GRANTED){
ActivityCompat.requestPermissions(MainActivity.this,new String[]{Manifest.permission.CALL_PHONE},1);
//如果没有授权的话,则需要调用ActivityCompat.requestPermissions()方法来向用户申请授权。
//requestPermissions接收三个参数,第一个参数要求是Activity的实例,第二个参数是一个String数组,我们要把申请的权限名放在数组中即可。
//第三个参数是请求码,只要是唯一值就可以了。这里传入1.
}else{
call();
//如果授权的话,直接执行拨打电话的逻辑操作。
}
}
});
}
private void call(){
try{
Intent intent = new Intent(Intent.ACTION_CALL);
intent.setData(Uri.parse("tel:10086"));
startActivity(intent);
}catch (SecurityException e){
e.printStackTrace();
}
}
//调用完了requestPermissions()方法之后,系统会弹出一个权限申请的对话框,然后用户可以选择同意或者拒绝。
//不论是哪种结果,最终都会调到onRequestPermissionsResult()方法中,而授权的结果则会封装在grantResults参数中。
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
switch(requestCode){
case 1:
//判断最后的授权结果,如果用户同意就调用call()方法来拨打电话。如果用户拒绝的话我们只能放弃操作,并弹出一条失败提示。
if (grantResults.length > 0 &&grantResults[0] == PackageManager.PERMISSION_GRANTED){
call();
}else{
Toast.makeText(this, "You denied the permission", Toast.LENGTH_SHORT).show();
}
break;
default:
}
}
}