Android 6.0以上版本權限管理


Android 6.0以上版本權限管理(targetSdkVision >= 23)

簡述

  Android 6.0之前在AndroidManifest中聲明可能用到的所有權限,用戶在安裝時,系統展示所有權限,用戶安裝即授予所有權限,取消則拒絕安裝。安裝后,應用就可以在用戶不知情的情況下進行非法操作(比如偷偷的上傳用戶數據)。

  為了更好地保護用戶隱私,Android 6.0將權限分為一般權限和危險權限兩種:(1)Normal Permissions,即普通權限,這類權限不會潛藏有侵害用戶隱私和安全的問題,比如,訪問網絡的權限,訪問WIFI的權限等。這類權限跟以前一樣在AndroidManifest聲明即可;(2)Dangerous Permissions,即危險權限,這類權限會直接的威脅到用戶的安全和隱私問題,比如說訪問短信,相冊等權限。這類危險權限需要開發者在代碼中手動的動態申請。

  同時,普通權限是單條的權限,而危險權限是以組展示的,如下就顯示了危險權限種類,一旦獲取一個危險權限,那么這一組危險權限都會被獲取。

危險權限有:

SMS(短信)

  1.SEND_SMS  2.RECEIVE_SMS  3.READ_SMS  4.RECEIV\E_WAP_PUSH  5.RECEIVE_MMS

STORAGE(存儲卡)

  1.READ_EXTERNAL_STORAGE  2.WRITE_EXTERNAL_STORAGE

CONTACTS(聯系人)

  1.READ_CONTACTS  2.WRITE_CONTACTS  3.GET_ACCOUNTS

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

危險權限申請方法

1.AndroidMinifest.xml中同樣要申請;

2.軟件運行時申請

  主要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.處理權限回調

首先定義一個Acyivity

  這個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()后則會退出程序。

  附:這里附上兩篇退出應用程序的文章,可以參考參考。

    1.Android-完全退出當前應用程序的四種方法

    2.android——徹底關閉——應用程序


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM