一、權限處理分類
由上圖可以看出,主要分為四類。下表逐一介紹各類對應的一些情況。
二、動態權限申請
雖然總的來說分為四類,但是只需要處理一種情況,即動態申請權限。其他三種情況,要么默認實現,要么系統定制,無法從代碼角度進行調整。那么下面先來看下那些權限需要動態申請。
(1)權限列表
Android6.0以上把權限分為普通權限和危險權限,所以危險權限是需要動態申請,給予用戶提示的,而危險權限就是上表展示的內容。
看到上面的 permissions,會發現一個問題,危險權限都是一組一組的。
分組對權限機制的申請是有一定影響的。例如app運行在android 6.x的機器上,對於授權機制是這樣的。如果你申請某個危險的權限,假設你的app早已被用戶授權了同一組的某個危險權限,那么系統會立即授權,而不需要用戶去點擊授權。比如你的app對READ_CONTACTS已經授權了,當你的app申請WRITE_CONTACTS時,系統會直接授權通過。
此外,對於申請時的彈窗上面的文本說明也是對整個權限組的說明,而不是單個權限。
下面介紹下Android 6.0以上 動態申請權限所設計到的一些方法。
(2)權限申請方法
在申請權限先,首先要保證在AndroidManifest中寫明需要的權限。 例如:
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"></uses-permission> <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"></uses-permission>
具體權限方法詳解:
權限申請示例
以獲取定位權限為例
1.點擊按鈕,檢查並申請權限
btn.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { if (Build.VERSION.SDK_INT >23) { if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.ACCESS_COARSE_LOCATION) == PackageManager.PERMISSION_GRANTED) { //授予權限 getLoation(); }else{ //未獲得權限 requestPermissions(new String[]{Manifest.permission.ACCESS_COARSE_LOCATION} ,REQUEST_CODE_LOCATION); } } } });
如果有權限,執行獲取位置邏輯,如果沒權限,則進行請求權限。
2.權限申請結果回調
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { if (requestCode == REQUEST_CODE_LOCATION) { if (grantResults[0] == PackageManager.PERMISSION_GRANTED) { getLoation(); } else { if (shouldShowRequestPermissionRationale( Manifest.permission.ACCESS_COARSE_LOCATION)){ new AlertDialog.Builder(this) .setMessage("申請定位權限,才能為你推送更准確的信息") .setPositiveButton("確定", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { //申請定位權限 requestPermissions(MainActivity.this, new String[]{Manifest.permission.ACCESS_COARSE_LOCATION}, REQUEST_CODE_LOCATION); } }).show(); } } return; } super.onRequestPermissionsResult(requestCode, permissions, grantResults); }
如果同意,執行獲取位置邏輯,如果拒絕,重寫shouldShowRequestPermissionRationale方法,返回true,向用戶彈窗給出一個獲取權限的提示,點擊后再次申請權限。
public boolean shouldShowRequestPermissionRationale(@NonNull String permission) { if (permission.equals(Manifest.permission.ACCESS_COARSE_LOCATION) ) { return true; } else { return super.shouldShowRequestPermissionRationale(permission); } }
重寫shouldShowRequestPermissionRationale,在申請位置權限時,返回true,給用戶解釋。
以上就是動態申請權限的邏輯,大概流程如下:
注意: shouldShowRequestPermissionRationale :默認情況下,不重寫該方法,在Android原生系統中,如果第二次彈出權限申請的對話框,會出現“以后不再彈出”的提示框,如果用戶勾選了,你再申請權限,則shouldShowRequestPermissionRationale返回true,意思是說要給用戶一個 解釋,告訴用戶為什么要這個權限。
轉載自:https://lrh1993.gitbooks.io/android_interview_guide/content/android/advance/permission.html