Android 運(yùn)行時(shí)權(quán)限

  1. M 6.0
  2. O 8.0

M 6.0


Android 6.0 版本(Api 23)推出了 動(dòng)態(tài)權(quán)限管理

應(yīng)用權(quán)限簡介

Android應(yīng)用默認(rèn)情況下不擁有任何權(quán)限,申請(qǐng)權(quán)限要在AndroidManifest.xml中靜態(tài)聲明。

若未在manifest中聲明權(quán)限具帮,則在調(diào)用相應(yīng)功能時(shí)蒜绽,將拋出異常,一般不會(huì)catch該異常包竹,程序會(huì)直接崩潰:
Caused by: java.lang.SecurityException: Permission denied (missing INTERNET permission?)

在** Android 6.0 之前, 所有的權(quán)限都在安裝應(yīng)用時(shí)顯示給用戶,選擇安裝則表示授權(quán)全部權(quán)限周瞎,之后無法取消授權(quán)苗缩。
Android 6.0 **之后, 危險(xiǎn)權(quán)限(dangerous)需在程序運(yùn)行時(shí)顯式彈框声诸,請(qǐng)求用戶授權(quán)酱讶。何時(shí)彈框由應(yīng)用程序決定。

對(duì)于普通權(quán)限(normal)双絮,仍保持原聲明方式浴麻,在安裝應(yīng)用程序時(shí)予以授權(quán)。

保護(hù)等級(jí)

permission的保護(hù)等級(jí)通過protectionLevel屬性設(shè)置囤攀,共4種:

  1. normal
  2. dangerous
  3. signature
  4. signatureOrSystem

詳見:http://developer.android.com/guide/topics/manifest/permission-element.html

簽名相關(guān)的不常用软免, 剩下normaldangerous
官網(wǎng) Guides: https://developer.android.com/guide/topics/security/permissions.html#normal-dangerous

總結(jié)下來:所有的權(quán)限仍在manifest中靜態(tài)聲明,normal權(quán)限在安裝時(shí)自動(dòng)授權(quán)焚挠,dangerous權(quán)限需應(yīng)用明確地請(qǐng)求用戶授權(quán)膏萧。
對(duì)于Android 6.0以下的手機(jī),或以前開發(fā)的舊應(yīng)用蝌衔,dangerous權(quán)限也是安裝時(shí)授權(quán)榛泛。

動(dòng)態(tài)權(quán)限流程圖
‘不再提示’流程圖

normal 類權(quán)限

當(dāng)安裝或更新時(shí),系統(tǒng)將授予應(yīng)用請(qǐng)求的屬于 PROTECTION_NORMAL 的所有權(quán)限噩斟。只需在AndroidManifest.xml中聲明曹锨,不必每次使用都檢查權(quán)限,且用戶不能取消以上授權(quán)剃允。

序號(hào) 權(quán)限
1 android.permission.ACCESS_LOCATION_EXTRA_COMMANDS
2 android.permission.ACCESS_NETWORK_STATE
3 android.permission.ACCESS_NOTIFICATION_POLICY
4 android.permission.ACCESS_WIFI_STATE
5 android.permission.ACCESS_WIMAX_STATE
6 android.permission.BLUETOOTH
7 android.permission.BLUETOOTH_ADMIN
8 android.permission.BROADCAST_STICKY
9 android.permission.CHANGE_NETWORK_STATE
10 android.permission.CHANGE_WIFI_MULTICAST_STATE
11 android.permission.CHANGE_WIFI_STATE
12 android.permission.CHANGE_WIMAX_STATE
13 android.permission.DISABLE_KEYGUARD
14 android.permission.EXPAND_STATUS_BAR
15 android.permission.FLASHLIGHT
16 android.permission.GET_ACCOUNTS
17 android.permission.GET_PACKAGE_SIZE
18 android.permission.INTERNET
19 android.permission.KILL_BACKGROUND_PROCESSES
20 android.permission.MODIFY_AUDIO_SETTINGS
21 android.permission.NFC
22 android.permission.READ_SYNC_SETTINGS
23 android.permission.READ_SYNC_STATS
24 android.permission.RECEIVE_BOOT_COMPLETED
25 android.permission.REORDER_TASKS
26 android.permission.REQUEST_INSTALL_PACKAGES
27 android.permission.SET_TIME_ZONE
28 android.permission.SET_WALLPAPER
29 android.permission.SET_WALLPAPER_HINTS
30 android.permission.SUBSCRIBED_FEEDS_READ
31 android.permission.TRANSMIT_IR
32 android.permission.USE_FINGERPRINT
33 android.permission.VIBRATE
34 android.permission.WAKE_LOCK
35 android.permission.WRITE_SYNC_SETTINGS
36 com.android.alarm.permission.SET_ALARM
37 com.android.launcher.permission.INSTALL_SHORTCUT
38 com.android.launcher.permission.UNINSTALL_SHORTCUT

權(quán)限組

新的權(quán)限模型提出了權(quán)限組的概念沛简,即:同權(quán)限組內(nèi)的某個(gè)權(quán)限被授權(quán)了,則該組中剩余的權(quán)限也會(huì)被自動(dòng)獲取授權(quán)斥废。例:Android.permission-group.CALENDAR權(quán)限組中的android.permission.WRITE_CALENDAR權(quán)限被授權(quán)椒楣,則應(yīng)用會(huì)自動(dòng)獲取android.permission.READ_CALENDAR權(quán)限。

序號(hào) 名稱 權(quán)限組 權(quán)限
1 日歷 android.permission-group.CALENDAR android.permission.READ_CALENDAR
android.permission.WRITE_CALENDAR
2 攝像頭 android.permission-group.CAMERA android.permission.CAMERA
3 通訊錄 android.permission-group.CONTACTS android.permission.READ_CONTACTS
android.permission.WRITE_CONTACTS
android.permission.GET_ACCOUNTS
4 地理位置 android.permission-group.LOCATION android.permission.ACCESS_FINE_LOCATION
android.permission.ACCESS_COARSE_LOCATION
5 麥克風(fēng) android.permission-group.MICROPHONE android.permission.RECORD_AUDIO
6 電話 android.permission-group.PHONE android.permission.READ_PHONE_STATE
android.permission.CALL_PHONE
android.permission.READ_CALL_LOG
android.permission.WRITE_CALL_LOG
com.android.voicemail.permission.ADD_VOICEMAIL
android.permission.USE_SIP
android.permission.PROCESS_OUTGOING_CALLS
7 傳感器 android.permission-group.SENSORS android.permission.BODY_SENSORS
8 短信 android.permission-group.SMS android.permission.SEND_SMS
android.permission.RECEIVE_SMS
android.permission.READ_SMS
android.permission.RECEIVE_WAP_PUSH
android.permission.RECEIVE_MMS
android.permission.READ_CELL_BROADCASTS
9 存儲(chǔ)空間 android.permission-group.STORAGE android.permission.READ_EXTERNAL_STORAGE
android.permission.WRITE_EXTERNAL_STORAGE

動(dòng)態(tài)權(quán)限使用

判斷是否有權(quán)限:checkSelfPermission()

如果沒有權(quán)限牡肉,彈窗給用戶選擇:requestPermission()捧灰,第二個(gè)參數(shù)codeonRequestPermissionResult()方法中的code對(duì)應(yīng)。

if(ContextCompat.checkSelfPermission(Manifest.permission.CAMERA) != PackageManager.PERMISSION_GRANTED) {  
    ActivityCompat.requestPermissions(new String[] { Manifest.permission.CAMERA}, REQUEST_CAMERA);  
}  

彈出權(quán)限選擇對(duì)話框前彈出提示统锤,用于引導(dǎo)用戶選擇:shouldShowRequestPermissionRationale()

if(ContextCompat.checkSelfPermission(Manifest.permission.CAMERA) != PackageManager.PERMISSION_GRANTED) {  
    if (ActivityCompat.shouldShowRequestPermissionRationale(activity.this, Manifest.permission.CAMERA)) {    
        Snackbar.make(view, "請(qǐng)?jiān)试S應(yīng)用使用照相機(jī)", Snackbar.LENGTH_INDEFINITE)            
            .setAction(R.string.ok, new View.OnClickListener() {                
                @Override               
                public void onClick(View view) {                      
                    ActivityCompat.requestPermissions(activity, new String[]{Manifest.permission.CAMERA}, REQUEST_CAMERA);                
                }            
            })            
          .show();
    } else {    
        ActivityCompat.requestPermissions(new String[] { Manifest.permission.CAMERA}, REQUEST_CAMERA);  
    }
}  

判斷用戶是否已確認(rèn)權(quán)限:onRequestPermissionResult()

@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions,                                       @NonNull int[] grantResults) { 
    switch (requestCode) {
        case REQUEST_CAMERA:
            if (PermissionUtil.verifyPermissions(grantResults)) { 
                Snackbar.make(mLayout, "同意授權(quán)", Snackbar.LENGTH_SHORT).show(); 
            } else { 
                Snackbar.make(mLayout, "拒絕授權(quán)", Snackbar.LENGTH_SHORT).show();                  
            }   
        break;  
        default: 
            super.onRequestPermissionsResult(requestCode, permissions, grantResults);    
    }
}

若用戶在選擇權(quán)限對(duì)話框時(shí)拒絕了某個(gè)權(quán)限的申請(qǐng)毛俏,那么再次申請(qǐng)?jiān)摍?quán)限時(shí)會(huì)多出一個(gè)“不再詢問”的checkbox,若勾選饲窿,則程序再調(diào)用requestPermission()拧抖,對(duì)話框也不會(huì)彈出。

同時(shí)處理多個(gè)權(quán)限免绿,方案待定

兼容問題

一. 動(dòng)態(tài)權(quán)限檢查需在android 6.0版本以上運(yùn)行,即api 23之前不能運(yùn)行:

  1. 判斷SDK版本號(hào)
public static boolean checkSDKVersion_23() {    
    return Build.VERSION.SDK_INT >= Build.VERSION_CODES.M;
}
  1. 使用v4兼容庫擦盾,兼容所有版本
  • ContextCompat.checkSelfPermission()
    授權(quán)返回PERMISSION_GRANTED嘲驾,否則返回PERMISSION_DENIED淌哟,所有版本都是如此。

  • ActivityCompat.requestPermissions()
    M之前版本調(diào)用辽故,OnRequestPermissionsResultCallback 直接被調(diào)用徒仓,帶著正確的 PERMISSION_GRANTEDPERMISSION_DENIED

  • ActivityCompat.shouldShowRequestPermissionRationale()
    M之前版本調(diào)用誊垢,永遠(yuǎn)返回false掉弛。

二. 在Fragment中使用,用v13兼容包喂走,效果一樣:

  • FragmentCompat.requestPermissions()
  • FragmentCompat.shouldShowRequestPermissionRationale()
  • Fragment中嵌套Fragment殃饿,子Fragment中建議用getParentFragment().requestPermissions方法回調(diào)父Fragment中的onRequestPermissionsResult,添加以下代碼將回調(diào)透傳到子Fragment
@Override 
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) { 
      super.onRequestPermissionsResult(requestCode, permissions, grantResults);     
      List<Fragment> fragmentList = getChildFragmentManager().getFragments(); 
      if (fragmentList != null && fragmentList.size > 0) { 
          for (Fragment fragment : fragmentList ) { 
              if (fragment != null) { 
                  fragment.onRequestPermissionsResult(requestCode,permissions,grantResults); 
              } 
          } 
      } 
}

開源

PermissionsDispatcher 使用標(biāo)注方式暫不支持嵌套Fragment
RxPermissions 基于RxJava
android-RuntimePermissions 谷歌官方示例

參考

Android M 動(dòng)態(tài)權(quán)限獲取

O 8.0


官方:
在 Android 8.0 之前芋肠,如果應(yīng)用在運(yùn)行時(shí)請(qǐng)求權(quán)限并且被授予該權(quán)限乎芳,系統(tǒng)會(huì)錯(cuò)誤地將屬于同一權(quán)限組并且在清單中注冊(cè)的其他權(quán)限也一起授予應(yīng)用。

對(duì)于針對(duì) Android 8.0 的應(yīng)用帖池,此行為已被糾正奈惑。系統(tǒng)只會(huì)授予應(yīng)用明確請(qǐng)求的權(quán)限。然而睡汹,一旦用戶為應(yīng)用授予某個(gè)權(quán)限肴甸,則所有后續(xù)對(duì)該權(quán)限組中權(quán)限的請(qǐng)求都將被自動(dòng)批準(zhǔn)。

例如囚巴,假設(shè)某個(gè)應(yīng)用在其清單中列出 READ_EXTERNAL_STORAGE原在。應(yīng)用請(qǐng)求 READ_EXTERNAL_STORAGE,并且用戶授予了該權(quán)限文兢。如果該應(yīng)用針對(duì)的是 API 級(jí)別 24 或更低級(jí)別晤斩,系統(tǒng)還會(huì)同時(shí)授予 WRITE_EXTERNAL_STORAGE,因?yàn)樵摍?quán)限也屬于同一 STORAGE 權(quán)限組并且也在清單中注冊(cè)過姆坚。如果該應(yīng)用針對(duì)的是 Android 8.0澳泵,則系統(tǒng)此時(shí)僅會(huì)授予 READ_EXTERNAL_STORAGE;不過兼呵,如果該應(yīng)用后來又請(qǐng)求 WRITE_EXTERNAL_STORAGE兔辅,則系統(tǒng)會(huì)立即授予該權(quán)限,而不會(huì)提示用戶击喂。

8.0 變更

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末维苔,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子懂昂,更是在濱河造成了極大的恐慌介时,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,331評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異沸柔,居然都是意外死亡循衰,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,372評(píng)論 3 398
  • 文/潘曉璐 我一進(jìn)店門褐澎,熙熙樓的掌柜王于貴愁眉苦臉地迎上來会钝,“玉大人,你說我怎么就攤上這事工三∏ㄋ幔” “怎么了?”我有些...
    開封第一講書人閱讀 167,755評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵俭正,是天一觀的道長奸鬓。 經(jīng)常有香客問我,道長段审,這世上最難降的妖魔是什么全蝶? 我笑而不...
    開封第一講書人閱讀 59,528評(píng)論 1 296
  • 正文 為了忘掉前任,我火速辦了婚禮寺枉,結(jié)果婚禮上抑淫,老公的妹妹穿的比我還像新娘。我一直安慰自己姥闪,他們只是感情好始苇,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,526評(píng)論 6 397
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著筐喳,像睡著了一般催式。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上避归,一...
    開封第一講書人閱讀 52,166評(píng)論 1 308
  • 那天荣月,我揣著相機(jī)與錄音,去河邊找鬼梳毙。 笑死哺窄,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的账锹。 我是一名探鬼主播萌业,決...
    沈念sama閱讀 40,768評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼奸柬!你這毒婦竟也來了生年?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,664評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤廓奕,失蹤者是張志新(化名)和其女友劉穎抱婉,沒想到半個(gè)月后档叔,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,205評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡蒸绩,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,290評(píng)論 3 340
  • 正文 我和宋清朗相戀三年蹲蒲,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片侵贵。...
    茶點(diǎn)故事閱讀 40,435評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖缘薛,靈堂內(nèi)的尸體忽然破棺而出窍育,到底是詐尸還是另有隱情,我是刑警寧澤宴胧,帶...
    沈念sama閱讀 36,126評(píng)論 5 349
  • 正文 年R本政府宣布漱抓,位于F島的核電站,受9級(jí)特大地震影響恕齐,放射性物質(zhì)發(fā)生泄漏乞娄。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,804評(píng)論 3 333
  • 文/蒙蒙 一显歧、第九天 我趴在偏房一處隱蔽的房頂上張望仪或。 院中可真熱鬧,春花似錦士骤、人聲如沸范删。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,276評(píng)論 0 23
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽到旦。三九已至,卻和暖如春巨缘,著一層夾襖步出監(jiān)牢的瞬間添忘,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,393評(píng)論 1 272
  • 我被黑心中介騙來泰國打工若锁, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留搁骑,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,818評(píng)論 3 376
  • 正文 我出身青樓拴清,卻偏偏與公主長得像靶病,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子口予,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,442評(píng)論 2 359

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