由上圖能夠看出,主要分爲四類。下表逐一介紹各種對應的一些狀況。android
雖然總的來講分爲四類,可是隻須要處理一種狀況,即動態申請權限。其餘三種狀況,要麼默認實現,要麼系統定製,沒法從代碼角度進行調整。那麼下面先來看下那些權限須要動態申請。bash
Android6.0以上把權限分爲普通權限和危險權限,因此危險權限是須要動態申請,給予用戶提示的,而危險權限就是上表展現的內容。app
看到上面的 permissions,會發現一個問題,危險權限都是一組一組的。ide
分組對權限機制的申請是有必定影響的。例如app運行在android 6.x的機器上,對於受權機制是這樣的。若是你申請某個危險的權限,假設你的app早已被用戶受權了同一組的某個危險權限,那麼系統會當即受權,而不須要用戶去點擊受權。好比你的app對READ_CONTACTS已經受權了,當你的app申請WRITE_CONTACTS時,系統會直接受權經過。ui
此外,對於申請時的彈窗上面的文本說明也是對整個權限組的說明,而不是單個權限。this
下面介紹下Android 6.0以上 動態申請權限所設計到的一些方法。spa
在申請權限先,首先要保證在AndroidManifest中寫明須要的權限。 例如:設計
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"></uses-permission>
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"></uses-permission>
複製代碼
具體權限方法詳解:3d
以獲取定位權限爲例。code
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,意思是說要給用戶一個 解釋,告訴用戶爲何要這個權限。