安卓系統的權限管理機制從API 23 (也就是Android 6.0 又叫做 Android M,)之後發生了比較大的改變,在一些比較危險的權限上要求必須申請動态權限,即使你在AndroidMainfest.xml檔案中申請也沒有任何用,或者你可以将編譯的目标版本設定這API 22,這樣就可以了。但這并不是長久之計,不是嗎?是以是以在這裡學習一下。
動态權限需求原因
- Android 6.0之前,權限在應用安裝過程中隻詢問一次,以清單的形式展現給使用者,然而大多數使用者并不會注意到這些,直接就下一步了,應用安裝成功後就會被賦予清單檔案中的所有權限,應用就可以在使用者不知情的情況下進行非法操作(比如偷偷的上傳使用者資料)。
需要動态申請的權限如下:
- READ_CALENDAR , WRITE_CALENDAR 讀寫月曆權限
- CAMERA 調用相機權限
- READ_CONTACTS , WRITE_CONTACTS , GET_ACCOUNTS 通訊錄權限
- ACCESS_FINE_LOCATION, ACCESS_COARSE_LOCATION 定位權限
- RECORD_AUDIO 錄音權限
- READ_PHONE_STATE ,CALL_PHONE READ_CALL_LOG, WRITE_CALL_LOG ADD_VOICEMAIL//手機狀态相關
- BODY_SENSORS 傳感器權限
-
SMS SEND_SMS ,RECEIVE_SMS ,READ_SMS, RECEIVE_WAP_PUSH, RECEIVE_MMS SMS消息權限
10.READ_EXTERNAL_STORAGE ,WRITE_EXTERNAL_STORAGE 外部存儲權限
動态權限申請步驟
AndroidMainfest.xml檔案配置
- 沒有具體分析是什麼原因,可能是為了相容低版本吧,有興趣的自行Google一下
判斷Android系統版本
- 在官方的文檔中,可以看到低于API23 是不需要使用動态權限申請的,我們需要判斷一下,代碼如下: 如果是Android 6.0以上的系統,需要進行判斷,我們大緻看一下
if (Build.VERSION.SDK_INT>=23) {
//此處做動态權限申請
}
else {
//低于23 不需要特殊處理
}
}
檢查權限
- 如果版本高于23 ,則需要進行特殊處理,我們這裡檢查一下有沒有權限
- 使用ContextCompat.CheckSlefPermission ,我們看看API解釋,從源代碼中找到這方法,我們來分析一下
/** * Determine whether <em>you</em> have been granted a particular permission.
* * @param permission The name of the permission being checked.
* * @return {@linkandroid.content.pm.PackageManager#PERMISSION_GRANTED} if you have the
* permission, or {@link android.content.pm.PackageManager#PERMISSION_DENIED} if not.
* * @see android.content.pm.PackageManager#checkPermission(String, String) */
public static int checkSelfPermission(@NonNull Context context, @NonNull String permission)
{
if (permission == null)
{
throw new IllegalArgumentException("permission is null");
}
return context.checkPermission(permission, android.os.Process.myPid(), Process.myUid());}
- 參數:一個上下文Context和一個權限的名稱。
-
傳回:PERMISSION_GRANTED 存在權限以及PERMISSION_DENIED 不存在權限
同時又調用了context.checkPermission方法,我們來看一下參數,拿到了一個String類型的資料,這個是權限的名稱嗎,以及目前程序的PID和UID,有興趣額的可以繼續深究
申請權限
- 如果發現CheckSelfPermission傳回值是 PERMISSION_DENIED,則需要進行權限申請,我們這裡看一下怎麼申請的
- 使用ContextCompat.CheckSlefPermission ,我們看看API解釋,從源代碼中找到這方法,我們來分析一下
- 參數說明: 目前上下文。一個權限數組,還有一個唯一的請求碼,注意這個請求碼要大于0 ,低于65535,因為程式要求請求碼隻能是16位的資料,被坑了一次
- 上下文就不說了,權限數組,說明是可以一次申請多個權限的,由于這個權限請求是異步操作的,是以說,使用者判斷權限後需要回調函數,那麼就用到這個請求碼了,好了,我們先來看看整體的邏輯代碼
if (Build.VERSION.SDK_INT>=23) {
int request=ContextCompat.checkSelfPermission(this, Manifest.permission.CAMERA);
if (request!= PackageManager.PERMISSION_GRANTED)//缺少權限,進行權限申請
{
ActivityCompat.requestPermissions(this,new String[]{Manifest.permission.CAMERA},825638);
return;//
}else
{
//權限同意
}
}
else
{ //低于23 不需要特殊處理,去掉用拍照的方法 }

回調函數的處理
- 由于程式是異步操作,在使用者完成了操作後,需要調用回調函數,而此回調函數則是一個Activity的放,是以重寫Activity的方法
//參數 requestCode是我們在申請權限的時候使用的唯一的申請碼
//String[] permission則是權限清單,一般用不到
//int[] grantResults 是使用者的操作響應,包含這權限是夠請求成功
//由于在權限申請的時候,我們就申請了一個權限,是以此處的數組的長度都是1
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[]
grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if (requestCode==123) {
//當然權限多了,建議使用Switch,不必糾結于此
if (grantResults[0]==PackageManager.PERMISSION_GRANTED)
{
Toast.makeText(this, "權限申請成功", Toast.LENGTH_SHORT).show();
}else if (grantResults[0]== PackageManager.PERMISSION_DENIED) {
Toast.makeText(this, "權限申請失敗,使用者拒絕權限", Toast.LENGTH_SHORT).show();
}
}
}
代碼套餐
- 代碼比較簡單,邊學邊寫的,收獲很大,以後要多寫BLOG了,比自己看一遍了解更深,記得更牢靠
package cn.wuhu.authority;
import android.Manifest;
import android.content.pm.PackageManager;import android.os.Build;
import android.support.annotation.NonNull;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.ContextCompat;
import android.support.v4.content.PermissionChecker;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;import android.view.View;
import android.widget.Toast;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
findViewById(R.id.button).setOnClickListener(new View.OnClickListener() { @Override
public void onClick(View view) {
onTakePhoto();
}
});
}
public void onTakePhoto() {
if (Build.VERSION.SDK_INT>=23) {
int request=ContextCompat.checkSelfPermission(this, Manifest.permission.CAMERA);
if (request!= PackageManager.PERMISSION_GRANTED)//缺少權限,進行權限申請
{
ActivityCompat.requestPermissions(this,new String[]{Manifest.permission.CAMERA},123);
return;//
}
else
{
//權限同意,不需要處理,去掉用拍照的方法 Toast.makeText(this,"權限同意",Toast.LENGTH_SHORT).show();
}
}
else{
//低于23 不需要特殊處理,去掉用拍照的方法
}
}
//參數 requestCode是我們在申請權限的時候使用的唯一的申請碼
//String[] permission則是權限清單,一般用不到
//int[] grantResults 是使用者的操作響應,包含這權限是夠請求成功
//由于在權限申請的時候,我們就申請了一個權限,是以此處的數組的長度都是1
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { super.onRequestPermissionsResult(requestCode, permissions, grantResults); if (requestCode==123)
{ //當然權限多了,建議使用Switch,不必糾結于此
if (grantResults[0]==PackageManager.PERMISSION_GRANTED) {
Toast.makeText(this, "權限申請成功",Toast.LENGTH_SHORT).show();
}else if (grantResults[0]== PackageManager.PERMISSION_DENIED) {
Toast.makeText(this, "權限申請失敗,使用者拒絕權限", Toast.LENGTH_SHORT).show();
}
}
}
}
轉載于:https://www.cnblogs.com/zhoutao825638/p/10382024.html