Android 6.0以前在AndroidManifest中聲明可能用到的全部權限,用戶在安裝時,系統展現全部權限,用戶安裝即授予全部權限,取消則拒絕安裝。安裝後,應用就能夠在用戶不知情的狀況下進行非法操做(好比偷偷的上傳用戶數據)。html
爲了更好地保護用戶隱私,Android 6.0將權限分爲通常權限和危險權限兩種:(1)Normal Permissions,即普通權限,這類權限不會潛藏有侵害用戶隱私和安全的問題,好比,訪問網絡的權限,訪問WIFI的權限等。這類權限跟之前同樣在AndroidManifest聲明便可;(2)Dangerous Permissions,即危險權限,這類權限會直接的威脅到用戶的安全和隱私問題,好比說訪問短信,相冊等權限。這類危險權限須要開發者在代碼中手動的動態申請。java
同時,普通權限是單條的權限,而危險權限是以組展現的,以下就顯示了危險權限種類,一旦獲取一個危險權限,那麼這一組危險權限都會被獲取。android
危險權限有:安全
SMS(短信)網絡
1.SEND_SMS 2.RECEIVE_SMS 3.READ_SMS 4.RECEIV\E_WAP_PUSH 5.RECEIVE_MMSapp
STORAGE(存儲卡)ide
1.READ_EXTERNAL_STORAGE 2.WRITE_EXTERNAL_STORAGEui
CONTACTS(聯繫人)this
1.READ_CONTACTS 2.WRITE_CONTACTS 3.GET_ACCOUNTSspa
PHONE(手機)
1.READ_PHONE_STATE 2.CALL_PHONE 3.READ_CALL_LOG 4.WRITE_CALL_LOG 5.ADD_VOICEMAIL 6.USE_SIP 7.PROCESS_OUTGOING_CALLS
CALENDAR(日曆)
1.READ_CALENDAR 2.WRITE_CALENDAR
CAMERA(相機)
1.CAMERA(並非何時都要設置)
LOCATION(位置)
1.ACCESS_FINE_LOCATION 2.ACCESS_COARSE_LOCATION
SENSORS(傳感器)
1.BODY_SENSORS
MICROPHONE(麥克風)
1.RECORD_AUDIO
主要API(Android 6.0新添加):(1)checkSelfPermission (2)requestPermissions
(1)checkSelfPermission(Context context, String permission);方法中有兩個參數,分別是上下文,以及所申請的權限;
(2)void requestPermissions(final Activity activity,final String[] permissions, final int requestCode);方法中須要三個參數,當前的activity,所申請的權限,能夠是多個,最後就是請求碼,既然有請求碼說明它會有一個回調,也就是咱們下面要講的處理回調。請求權限對話框是系統默認的,不是自定義設計的;
(3)經過重載onRequestPermissionsResult處理權限回調。
步驟:1.檢查是否擁有權限;2.假如沒有權限,則申請權限;3.處理權限回調
這個activity專門用於打開軟件時申請權限
activity_permission.xml,tools:context屬性說明這個xml關聯到PermissionActivity。
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" xmlns:tools="http://schemas.android.com/tools" tools:context="silen.com.apppicturetrans.PermissionActivity"> > </LinearLayout>
在AndroidMinifest中將該activity設置爲開機加載的activity,AndroidMinifest.xml文件中部分代碼以下
<activity android:name=".MainActivity" /> <activity android:name=".PermissionActivity"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity>
注:
android.intent.action.MAIN決定應用程序最早啓動的Activity
android.intent.category.LAUNCHER決定應用程序是否顯示在程序列表裏
若是存在多個activity都聲明瞭android.intent.action.MAIN與android.intent.category.LAUNCHER將會有多個圖標顯示在桌面上
PermissionActivity.java文件:
import android.Manifest; import android.content.DialogInterface; import android.content.Intent; import android.content.pm.PackageManager; import android.os.Bundle; import android.support.annotation.Nullable; import android.support.v4.app.ActivityCompat; import android.support.v4.content.ContextCompat; import android.support.v7.app.AlertDialog; import android.support.v7.app.AppCompatActivity; import android.util.Log; import static android.Manifest.permission.ACCESS_FINE_LOCATION; import static android.Manifest.permission.WRITE_EXTERNAL_STORAGE; public class PermissionActivity extends AppCompatActivity { @Override protected void onCreate( Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_permission); primissionAsk(); } private void primissionAsk() { Log.e("權限判斷", "開始" ); if(ContextCompat.checkSelfPermission(this, WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED|| ContextCompat.checkSelfPermission(this, ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED) { //兩種權限缺乏一種則要申請 //若果是第一次 new AlertDialog.Builder(this).setMessage("爲了保證應用正常運行,須要內存卡訪問權限以及位置信息獲取權限!") .setPositiveButton("容許", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialogInterface, int i) { ActivityCompat.requestPermissions(PermissionActivity.this, new String[]{WRITE_EXTERNAL_STORAGE, ACCESS_FINE_LOCATION}, 001); } }).show(); } else { Intent intent = new Intent(this,MainActivity.class); startActivity(intent); finish(); //(2) } } public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) { if(requestCode == 001) { if(grantResults[0] == PackageManager.PERMISSION_GRANTED &&grantResults[1] == PackageManager.PERMISSION_GRANTED) { Intent intent = new Intent(this,MainActivity.class); startActivity(intent); finish(); //(3) } else { finish(); //(1) //0表示正常退出,1表示異常退出 System.exit(0); } } } }
如代碼中所示,我新建了一個permissionAsk方法,在方法中首先利用checkSelfPermission檢查是否已經獲取內存讀寫權限與位置獲取權限(注意,如以前提到的,這裏一旦獲取權限就是一組),若是已經獲取,則執行else中語句,轉到Mainactivity;若是沒有獲取權限,則顯示提示框AlertDialog向用戶說明"爲了保證應用正常運行,須要內存卡訪問權限以及位置信息獲取權限!",用戶點擊"容許"後將調用requestPermissions命令獲取權限(該對話框系統默認),分別獲取內存讀寫與位置權限,獲取後執行onRequestPermissionsResult命令轉到Mainactivity。
注:在上圖代碼中,若是沒有(1)處finish(),則程序沒法像想象中的那樣退出,只是黑屏後從新加載應用程序(具體緣由我也沒弄清楚,彷佛不少方法都很難乾淨的退出應用程序)。(2)與(3)處若是沒有finish()則在MainActivity中按返回鍵會返回PermissionActivity的空白界面,加上finish()後則會退出程序。
附:這裏附上兩篇退出應用程序的文章,能夠參考參考。