Android權限之動態權限


安卓系統的權限管理機制從API 23 (也就是Android 6.0 又叫做 Android M,)之后發生了比較大的改變,在一些比較危險的權限上要求必須申請動態權限,即使你在AndroidMainfest.xml文件中申請也沒有任何用,或者你可以將編譯的目標版本設定這API 22,這樣就可以了。但這並不是長久之計,不是嗎?所以因此在這里學習一下。

動態權限需求原因

  • Android 6.0之前,權限在應用安裝過程中只詢問一次,以列表的形式展現給用戶,然而大多數用戶並不會注意到這些,直接就下一步了,應用安裝成功后就會被賦予清單文件中的所有權限,應用就可以在用戶不知情的情況下進行非法操作(比如偷偷的上傳用戶數據)。

需要動態申請的權限如下:

  1. READ_CALENDAR , WRITE_CALENDAR 讀寫日歷權限
  2. CAMERA 調用相機權限
  3. READ_CONTACTS , WRITE_CONTACTS , GET_ACCOUNTS 通訊錄權限
  4. ACCESS_FINE_LOCATION, ACCESS_COARSE_LOCATION 定位權限
  5. RECORD_AUDIO 錄音權限
  6. READ_PHONE_STATE ,CALL_PHONE READ_CALL_LOG, WRITE_CALL_LOG ADD_VOICEMAIL//手機狀態相關
  7. BODY_SENSORS 傳感器權限
  8. SMS SEND_SMS ,RECEIVE_SMS ,READ_SMS, RECEIVE_WAP_PUSH, RECEIVE_MMS SMS消息權限
    10.READ_EXTERNAL_STORAGE ,WRITE_EXTERNAL_STORAGE 外部存儲權限

動態權限申請步驟

AndroidMainfest.xml文件配置

  • 沒有具體分析是什么原因,可能是為了兼容低版本吧,有興趣的自行Google一下

判斷Android系統版本

  • 在官方的文檔中,可以看到低於API23 是不需要使用動態權限申請的,我們需要判斷一下,代碼如下: 如果是Android 6.0以上的系統,需要進行判斷,我們大致看一下
    if (Build.VERSION.SDK_INT>=23) {        
            //此處做動態權限申請    
    }
  else {      
        //低於23 不需要特殊處理   
     }
  }

檢查權限

  • 如果版本高於23 ,則需要進行特殊處理,我們這里檢查一下有沒有權限
  • 使用ContextCompat.CheckSlefPermission ,我們看看API解釋,從源代碼中找到這方法,我們來分析一下
/** * Determine whether <em>you</em> have been granted a particular permission.
* * @param permission The name of the permission being checked. 
* * @return {@linkandroid.content.pm.PackageManager#PERMISSION_GRANTED} if you have the 
* permission, or {@link android.content.pm.PackageManager#PERMISSION_DENIED} if not. 
* * @see android.content.pm.PackageManager#checkPermission(String, String) */
public static int checkSelfPermission(@NonNull Context context, @NonNull String permission) 
{  
  if (permission == null)
     {  
        throw new IllegalArgumentException("permission is null");  
     }   
 return context.checkPermission(permission, android.os.Process.myPid(), Process.myUid());}
  • 參數:一個上下文Context和一個權限的名稱。
  • 返回:PERMISSION_GRANTED 存在權限以及PERMISSION_DENIED 不存在權限
    同時又調用了context.checkPermission方法,我們來看一下參數,拿到了一個String類型的數據,這個是權限的名稱嗎,以及當前進程的PID和UID,有興趣額的可以繼續深究

申請權限

  • 如果發現CheckSelfPermission返回值是 PERMISSION_DENIED,則需要進行權限申請,我們這里看一下怎么申請的
  • 使用ContextCompat.CheckSlefPermission ,我們看看API解釋,從源代碼中找到這方法,我們來分析一下
ActivityCompat.requestPermissions(thisActivity, new String[]{Manifest.permission.READ_CONTACTS}, REQUEST_CODE);
  • 參數說明: 當前上下文。一個權限數組,還有一個唯一的請求碼,注意這個請求碼要大於0 ,低於65535,因為程序要求請求碼只能是16位的數據,被坑了一次
  • 上下文就不說了,權限數組,說明是可以一次申請多個權限的,由於這個權限請求是異步操作的,所以說,用戶判斷權限后需要回調函數,那么就用到這個請求碼了,好了,我們先來看看整體的邏輯代碼
    if (Build.VERSION.SDK_INT>=23) {   
     int request=ContextCompat.checkSelfPermission(this, Manifest.permission.CAMERA);      
        if (request!= PackageManager.PERMISSION_GRANTED)//缺少權限,進行權限申請      
          {            
              ActivityCompat.requestPermissions(this,new String[]{Manifest.permission.CAMERA},825638);            
               return;//    
       }else 
           {       
             //權限同意
          }

      }
  else
      { //低於23 不需要特殊處理,去掉用拍照的方法  }

Screenshot_20161216-103812.png

回調函數的處理

  • 由於程序是異步操作,在用戶完成了操作后,需要調用回調函數,而此回調函數則是一個Activity的放,因此重寫Activity的方法

//參數 requestCode是我們在申請權限的時候使用的唯一的申請碼
//String[] permission則是權限列表,一般用不到

//int[] grantResults 是用戶的操作響應,包含這權限是夠請求成功

//由於在權限申請的時候,我們就申請了一個權限,所以此處的數組的長度都是1

@Override

public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] 
grantResults) {  

  super.onRequestPermissionsResult(requestCode, permissions, grantResults);  

  if (requestCode==123)    {     

   //當然權限多了,建議使用Switch,不必糾結於此  
 
     if (grantResults[0]==PackageManager.PERMISSION_GRANTED) 

       {

            Toast.makeText(this, "權限申請成功", Toast.LENGTH_SHORT).show(); 

       }else if (grantResults[0]== PackageManager.PERMISSION_DENIED)        { 

           Toast.makeText(this, "權限申請失敗,用戶拒絕權限", Toast.LENGTH_SHORT).show(); 

       } 

   }

}

Screenshot_20161216-103815.png
Screenshot_20161216-103928.png

代碼套餐

  • 代碼比較簡單,邊學邊寫的,收獲很大,以后要多寫BLOG了,比自己看一遍理解更深,記得更牢靠
package cn.wuhu.authority;
import android.Manifest;
import android.content.pm.PackageManager;import android.os.Build;
import android.support.annotation.NonNull;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.ContextCompat;
import android.support.v4.content.PermissionChecker;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;import android.view.View;
import android.widget.Toast;
public class MainActivity extends AppCompatActivity {    
@Override
 protected void onCreate(Bundle savedInstanceState) {        
super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main); 
       findViewById(R.id.button).setOnClickListener(new View.OnClickListener() {            @Override
            public void onClick(View view) {
                onTakePhoto();
            }
        });
    }
   public void onTakePhoto()   { 
      if (Build.VERSION.SDK_INT>=23)       {
           int request=ContextCompat.checkSelfPermission(this, Manifest.permission.CAMERA);
           if (request!= PackageManager.PERMISSION_GRANTED)//缺少權限,進行權限申請
           { 
              ActivityCompat.requestPermissions(this,new String[]{Manifest.permission.CAMERA},123);
                return;//
           }
           else
           {
               //權限同意,不需要處理,去掉用拍照的方法               Toast.makeText(this,"權限同意",Toast.LENGTH_SHORT).show();
           }
       }
  else{
           //低於23 不需要特殊處理,去掉用拍照的方法
       }
   }
    //參數 requestCode是我們在申請權限的時候使用的唯一的申請碼 
   //String[] permission則是權限列表,一般用不到
    //int[] grantResults 是用戶的操作響應,包含這權限是夠請求成功
    //由於在權限申請的時候,我們就申請了一個權限,所以此處的數組的長度都是1
    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {        super.onRequestPermissionsResult(requestCode, permissions, grantResults);        if (requestCode==123) 
       {            //當然權限多了,建議使用Switch,不必糾結於此
            if (grantResults[0]==PackageManager.PERMISSION_GRANTED)            {
                Toast.makeText(this, "權限申請成功",Toast.LENGTH_SHORT).show();
            }else if (grantResults[0]== PackageManager.PERMISSION_DENIED)            { 
               Toast.makeText(this, "權限申請失敗,用戶拒絕權限", Toast.LENGTH_SHORT).show();
            }
        } 
   }
}



免責聲明!

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



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