深入理解android6.0 RunTime Permisstion

What?

了解下runtime permission

2015.8 google發(fā)布了android 6.0,sdk版本為23,一款"為工作升級而生"的android系統(tǒng).如6.0新加入的指紋識別;Doze電量管理;快速充電切換...
還是說本文的重點吧,運行時權(quán)限,為了避免一些惡意app行為,如后臺流量偷跑,偷偷扣費等情況,google對安全做了進一步的整理和優(yōu)化.

對比android6.0之前有什么區(qū)別

  • targetSdkVersion 23以下時
    對于權(quán)限主需要在安裝時被詢問一次,而且是批量處理的,對于客戶而言一般都是很少仔細去看權(quán)限的風險內(nèi)容,
    直接安裝的,即使在對一些危險權(quán)限有紅色提醒.但是慣性的操作也解決不了,安全問題.

    3815189.png

  • targetSdkVersion 23以上時
    對于危險權(quán)限是需要單獨處理的,app在運行時只要接觸了危險權(quán)限,就會彈窗提醒,詢問用戶是否授權(quán).

    Paste_Image.png

  • 權(quán)限管理
    當然你也可以在setting - apps - xxApp - permissions中手動開啟和關(guān)閉對應權(quán)限.

    Paste_Image.png

6.0對權(quán)限的劃分

在整個權(quán)限列表內(nèi),權(quán)限可以分為normal,dangerous,special類型其實special也屬于dangerous類型,但是他的請求方式需要通過,
隱式意圖來處理,下面是微信權(quán)限和特殊權(quán)限的列表

  • dangerous permission(危險權(quán)限)


    Paste_Image.png
  • special permission(特殊權(quán)限)
    需要通過隱式意圖來開啟
    WRITE_SETTINGS
    SYSTEM_ALERT_WINDOW


Why?

runtime permission的出現(xiàn)主要解決什么問題?

google的更新主要歸納三點:性能的提示,信息的安全,規(guī)范的統(tǒng)一.而這次的運行時權(quán)限的更新主要就是對信息安全的處理,如6.0之前開發(fā)者在AndroidManifest清單文件上申請的權(quán)限會被系統(tǒng)默認授權(quán),然而用戶如果授權(quán)后想反悔取消這些授權(quán),就得通過第三方軟件來處理,這樣的方式既麻煩也很流氓,還有
比如特殊權(quán)限懸浮窗,如果一些開發(fā)者利用默認授權(quán)的方式,讓app一直開啟浮窗,這樣的體驗用戶也是不買單的,因此就出現(xiàn)了6.0的runtime permission.


How?

什么時候會開啟runtime permission设易?

  • app的gradle配置要求targetSdkVersion 23
compileSdkVersion 23
    buildToolsVersion "23.0.2"
    defaultConfig {
        targetSdkVersion 23
      ...
    }
  • 清單文件配置
    只有在涉及到危險權(quán)限時才會彈窗運行時權(quán)限請求
 <uses-permission android:name="android.permission.READ_PHONE_STATE"/>

google對涉及到危險權(quán)限是怎么處理的呢.

  • 檢查當前 targetSdk是否大于等于23
 private boolean isMNC() {
int targetSdkVersion =        context.getApplicationInfo().targetSdkVersion;
        return targetSdkVersion >= 23;
 }
  • 檢查是否需要使用到 READ_PHONE_STATE 權(quán)限,如果清單有配置則彈窗詢問授權(quán)
  int state = ContextCompat.checkSelfPermission(this,
                    Manifest.permission.READ_PHONE_STATE);
  • 申請 READ_PHONE_STATE 權(quán)限
 ActivityCompat.requestPermissions(this, new String[]{
                                Manifest.permission.WRITE_EXTERNAL_STORAGE
                                , Manifest.permission.READ_PHONE_STATE},
                        READ_PHONE_STATE_REQUEST_CODE);
  • 請求運行時權(quán)限r(nóng)equestPermissions 回調(diào)
 /**
     * 請求運行時權(quán)限r(nóng)equestPermissions 回調(diào)
     *
     * @param requestCode 請求碼
     * @param permissions 權(quán)限數(shù)組
     * @param grantResults 返回權(quán)限授權(quán)狀態(tài)數(shù)組結(jié)果, 0為已授權(quán)
     */
    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        if (requestCode == READ_PHONE_STATE_REQUEST_CODE) {
            if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                // Permission Granted
            } else {
                // Permission Denied
            }
        }
    }

</p><a >這是github上的Demo</a>


使用第三方RxPermisstion 處理

  • RxPermisstion處理方式
    批量處理
RxPermissions.getInstance(this)
    .request(Manifest.permission.READ_PHONE_STATE//DO
            , Manifest.permission.READ_CONTACTS//DO
            , Manifest.permission.GET_ACCOUNTS
            , Manifest.permission.WRITE_CONTACTS
            , Manifest.permission.ACCESS_FINE_LOCATION//DO
            , Manifest.permission.ACCESS_COARSE_LOCATION
            , Manifest.permission.WRITE_EXTERNAL_STORAGE
            , Manifest.permission.READ_EXTERNAL_STORAGE
            , Manifest.permission.SEND_SMS//DO
            , Manifest.permission.READ_SMS
            , Manifest.permission.RECEIVE_SMS
            , Manifest.permission.CAMERA)//DO
    .subscribe(isGranted -> {
        if (isGranted) {
            System.out.println("全已授權(quán)");
        
            doNext(true);
        } else {
       
            System.out.println("沒全授權(quán)");
        }
    });
  • 檢查這些權(quán)限中,有哪些被拒絕,授權(quán)
RxPermissions.getInstance(this)
        .requestEach(Manifest.permission.READ_PHONE_STATE
                , Manifest.permission.READ_CONTACTS
                , Manifest.permission.GET_ACCOUNTS
                , Manifest.permission.WRITE_CONTACTS
                , Manifest.permission.ACCESS_FINE_LOCATION
                , Manifest.permission.ACCESS_COARSE_LOCATION
                , Manifest.permission.WRITE_EXTERNAL_STORAGE
                , Manifest.permission.READ_EXTERNAL_STORAGE
                , Manifest.permission.SEND_SMS
                , Manifest.permission.READ_SMS
                , Manifest.permission.RECEIVE_SMS
                , Manifest.permission.CAMERA)
        .subscribe(permission -> {
            if (!permission.granted) {
                System.out.println("denied:" + permission.name);
                SToast.l(mAct, "部分權(quán)限未授權(quán),可能會導致app無法正常運行");
            } else {
                System.out.println("granted:" + permission.name);
            }
        });

在一些特殊權(quán)限下需要使用隱式詢問授權(quán)

  • 這些特殊權(quán)限也屬于危險權(quán)限,但是他們的授權(quán)方式與運行時權(quán)限不一樣需要使用隱式意圖開授權(quán).
//運行時權(quán)限所需求的彈窗,這邊需要先開啟運行彈窗權(quán)限
@TargetApi(Build.VERSION_CODES.M)
public static void requestAlertPermis(Context mcont, int requestCode) {
    Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION);
    intent.setData(Uri.parse("package:" + mcont.getPackageName()));
    ((Activity) mcont).startActivityForResult(intent, requestCode);
}
//如果項目使用到了推送就需要用到,Write_Settings權(quán)限,而它也是屬于特殊權(quán)限,因此需要隱式開啟授權(quán)
@TargetApi(Build.VERSION_CODES.M)
public static void requestSettingsPermis(Context mcont, int requestCode) {
    Intent intent = new Intent(Settings.ACTION_MANAGE_WRITE_SETTINGS);
    intent.setData(Uri.parse("package:" + mcont.getPackageName()));
    ((Activity) mcont).startActivityForResult(intent, requestCode);
}
  • 特殊權(quán)限的回調(diào)處理
 @Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if (requestCode == ALEAR_WINDOWS_REQUEST_CODE) {
            if (Settings.canDrawOverlays(mcont)) {//判斷是否開啟彈窗
               //TODO 請求6.0 所需要的運行時權(quán)限
                Toast.l(this, "彈窗權(quán)限已開啟逗柴!");
            } else {
                Toast.l(this, "請開啟彈窗權(quán)限!");
            }
        } else if (requestCode == WRITE_SETTINGS_REQUEST_CODE) {
           if(Settings.System.canWrite(mcont)){//判斷是否開啟修改系統(tǒng)
             //TODO 初始化推送
           }else {
              //TODO 其他你想要的處理
            } 
        }
}

實踐中遇到的坑

1異常信息:
You cannot keep your settings in the secure settings.
**原因: **
該異常是由于百度推送Writing_Settings權(quán)限引起的,由于android6.0對于一些權(quán)限需要特殊處理,如dangerous permission,special permission,因此在初始化百度推送時,需要先開啟Writing_Settings這個特殊權(quán)限,然而開啟后還是運行部了,原因是百度推送在sdk4.53以下版本,運行在android target為 23時有個坑,也就是這個異常.
解決方法: **
升級百度推送為4.6以上的sdk就可以解決了.
上面有特殊權(quán)限處理的代碼

2異常信息:
在AndroidManifest.xml中如果沒有配置運行時權(quán)限,直接調(diào)用requestPermissionsapp會崩潰
**解決方法: **
請求的運行時權(quán)限,必須要先在AndroidManifest.xml中配置.

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末亡嫌,一起剝皮案震驚了整個濱河市嚎于,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌挟冠,老刑警劉巖于购,帶你破解...
    沈念sama閱讀 216,591評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異知染,居然都是意外死亡肋僧,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,448評論 3 392
  • 文/潘曉璐 我一進店門控淡,熙熙樓的掌柜王于貴愁眉苦臉地迎上來嫌吠,“玉大人,你說我怎么就攤上這事掺炭”枳纾” “怎么了?”我有些...
    開封第一講書人閱讀 162,823評論 0 353
  • 文/不壞的土叔 我叫張陵涧狮,是天一觀的道長炕矮。 經(jīng)常有香客問我,道長者冤,這世上最難降的妖魔是什么肤视? 我笑而不...
    開封第一講書人閱讀 58,204評論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮涉枫,結(jié)果婚禮上邢滑,老公的妹妹穿的比我還像新娘。我一直安慰自己愿汰,他們只是感情好困后,可當我...
    茶點故事閱讀 67,228評論 6 388
  • 文/花漫 我一把揭開白布乐纸。 她就那樣靜靜地躺著,像睡著了一般摇予。 火紅的嫁衣襯著肌膚如雪锯仪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,190評論 1 299
  • 那天趾盐,我揣著相機與錄音庶喜,去河邊找鬼。 笑死救鲤,一個胖子當著我的面吹牛久窟,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播本缠,決...
    沈念sama閱讀 40,078評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼斥扛,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了丹锹?” 一聲冷哼從身側(cè)響起稀颁,我...
    開封第一講書人閱讀 38,923評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎楣黍,沒想到半個月后匾灶,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,334評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡租漂,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,550評論 2 333
  • 正文 我和宋清朗相戀三年阶女,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片哩治。...
    茶點故事閱讀 39,727評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡秃踩,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出业筏,到底是詐尸還是另有隱情憔杨,我是刑警寧澤,帶...
    沈念sama閱讀 35,428評論 5 343
  • 正文 年R本政府宣布蒜胖,位于F島的核電站消别,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏翠勉。R本人自食惡果不足惜妖啥,卻給世界環(huán)境...
    茶點故事閱讀 41,022評論 3 326
  • 文/蒙蒙 一霉颠、第九天 我趴在偏房一處隱蔽的房頂上張望对碌。 院中可真熱鬧,春花似錦蒿偎、人聲如沸朽们。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,672評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽骑脱。三九已至菜枷,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間叁丧,已是汗流浹背啤誊。 一陣腳步聲響...
    開封第一講書人閱讀 32,826評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留拥娄,地道東北人蚊锹。 一個月前我還...
    沈念sama閱讀 47,734評論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像稚瘾,于是被迫代替她去往敵國和親牡昆。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,619評論 2 354

推薦閱讀更多精彩內(nèi)容