1. 程式人生 > >談談Android 6.0 的動態許可權管理

談談Android 6.0 的動態許可權管理

1.前言

大家都知道Android 6.0的新特性之一就是應用許可權的管理。也就是說凡是涉及使用者隱私的許可權,使用者可以自己去設定管理了。然而在6.0以前,我們安裝一款APP是默認同意此APP所需的所有許可權(比如定位、訪問通訊錄),不同意就不能安裝。當然,國內的一些手機廠商基於Android定製的系統中,可以實現在6.0以前關閉指定的許可權。如下圖:

這裡寫圖片描述

這裡寫圖片描述

2.危險許可權列表(Dangerous Permission)

Dangerous Permission一般都是涉及使用者隱私的許可權。

這裡寫圖片描述

從上面的圖片中可以看到,攝像頭、電話、定位等等都是我們平常開發中常用的許可權。

3.可以在6.0不適配許可權管理嗎?

答案是可以,但是不推薦。

首先說怎麼不適配,那就是設定targetSdkVersion小於23(Android 6.0系統預設為targetSdkVersion小於23的應用預設授予了所申請的所有許可權,所以如果您APP設定的targetSdkVersion低於23,在執行時也不會崩潰。)

有人一看這不是挺好的嘛,解決問題。那麼我想告訴你,首先這不是長久之計,早晚都要面對的。你不可能永遠targetSdkVersion低於23。其次,它是有一個前提,那就是使用者自己不去操作許可權。要知道如果使用者是6.0以上的手機或是國內部分6.0以前的手機,他可以自己在設定中關閉許可權,那麼到時APP因為沒有許可權獲取資料異常,導致空指標的異常時,APP就會崩潰。

4.怎麼適配

首先Android Studio

在build.gradle中宣告targetSdkVersion為23及以上。

Eclipse

在AndroidManifest.xml中宣告targetSdkVersion為23及以上。

這裡引用高德定位Demo的CheckPermissionsActivity類,程式碼如下:

/**
 * 繼承了Activity,實現Android6.0的執行時許可權檢測
 * 需要進行執行時許可權檢測的Activity可以繼承這個類
 * 
 * @建立時間:2016年5月27日 下午3:01:31 
 * @專案名稱: AMapLocationDemo
 * @author
hongming.wang * @檔名稱:PermissionsChecker.java * @型別名稱:PermissionsChecker * @since 2.5.0 */
public class CheckPermissionsActivity extends Activity { /** * 需要進行檢測的許可權陣列 */ protected String[] needPermissions = { Manifest.permission.ACCESS_COARSE_LOCATION, Manifest.permission.ACCESS_FINE_LOCATION, Manifest.permission.WRITE_EXTERNAL_STORAGE, Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.READ_PHONE_STATE }; private static final int PERMISSON_REQUESTCODE = 0; /** * 判斷是否需要檢測,防止不停的彈框 */ private boolean isNeedCheck = true; @Override protected void onResume() { super.onResume(); if (Build.VERSION.SDK_INT >= 23 && getApplicationInfo().targetSdkVersion >= 23) { if (isNeedCheck) { checkPermissions(needPermissions); } } } /** * * @param permissions * @since 2.5.0 * requestPermissions方法是請求某一許可權, */ private void checkPermissions(String... permissions) { try { if (Build.VERSION.SDK_INT >= 23 && getApplicationInfo().targetSdkVersion >= 23) { List<String> needRequestPermissonList = findDeniedPermissions(permissions); if (null != needRequestPermissonList && needRequestPermissonList.size() > 0) { String[] array = needRequestPermissonList.toArray(new String[needRequestPermissonList.size()]); Method method = getClass().getMethod("requestPermissions", new Class[]{String[].class, int.class}); method.invoke(this, array, PERMISSON_REQUESTCODE); } } } catch (Throwable e) { } } /** * 獲取許可權集中需要申請許可權的列表 * * @param permissions * @return * @since 2.5.0 * checkSelfPermission方法是在用來判斷是否app已經獲取到某一個許可權 * shouldShowRequestPermissionRationale方法用來判斷是否 * 顯示申請許可權對話方塊,如果同意了或者不在詢問則返回false */ private List<String> findDeniedPermissions(String[] permissions) { List<String> needRequestPermissonList = new ArrayList<String>(); if (Build.VERSION.SDK_INT >= 23 && getApplicationInfo().targetSdkVersion >= 23){ try { for (String perm : permissions) { Method checkSelfMethod = getClass().getMethod("checkSelfPermission", String.class); Method shouldShowRequestPermissionRationaleMethod = getClass().getMethod("shouldShowRequestPermissionRationale", String.class); if ((Integer)checkSelfMethod.invoke(this, perm) != PackageManager.PERMISSION_GRANTED || (Boolean)shouldShowRequestPermissionRationaleMethod.invoke(this, perm)) { needRequestPermissonList.add(perm); } } } catch (Throwable e) { } } return needRequestPermissonList; } /** * 檢測是否所有的許可權都已經授權 * @param grantResults * @return * @since 2.5.0 * */ private boolean verifyPermissions(int[] grantResults) { for (int result : grantResults) { if (result != PackageManager.PERMISSION_GRANTED) { return false; } } return true; } /** * 申請許可權結果的回撥方法 */ @TargetApi(23) public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] paramArrayOfInt) { if (requestCode == PERMISSON_REQUESTCODE) { if (!verifyPermissions(paramArrayOfInt)) { showMissingPermissionDialog(); isNeedCheck = false; } } } /** * 顯示提示資訊 * * @since 2.5.0 * */ private void showMissingPermissionDialog() { AlertDialog.Builder builder = new AlertDialog.Builder(this); builder.setTitle(R.string.notifyTitle); builder.setMessage(R.string.notifyMsg); // 拒絕, 退出應用 builder.setNegativeButton(R.string.cancel, new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { finish(); } }); builder.setPositiveButton(R.string.setting, new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { startAppSettings(); } }); builder.setCancelable(false); builder.show(); } /** * 啟動應用的設定 * * @since 2.5.0 * */ private void startAppSettings() { Intent intent = new Intent( Settings.ACTION_APPLICATION_DETAILS_SETTINGS); intent.setData(Uri.parse("package:" + getPackageName())); startActivity(intent); } @Override public boolean onKeyDown(int keyCode, KeyEvent event) { if(keyCode == KeyEvent.KEYCODE_BACK){ this.finish(); return true; } return super.onKeyDown(keyCode, event); } }

我在上面的類中,自己加入了一些註釋,大家仔細看就可以明白了。

補充:小米手機在動態許可權這裡還需要一些相容,我們需要注意一下。當然對於國內部分6.0以前手機,只能在需要許可權去去捕獲異常來處理了。

當然不止上面一種實現方法,github上有許多大神開源的封裝庫,可以很方便的實現許可權適配。我推薦兩個庫,大家根據需求選擇:

5.參考