zoukankan      html  css  js  c++  java
  • 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();
                }
            } 
       }
    }
    
    
    
  • 相关阅读:
    性能测试_LR11_数据向导
    python django部署linux服务器
    pip升级问题
    解决win10命令提示行下cnpm无反应
    adb常用命令连接设备/查看包名/查看activity
    android虚拟机sdcard操作出现mkdir failed for , Read-only file system的解决办法
    Django项目和应用创建
    mac下配置python虚拟环境
    这个是我的标题_2020_01_01_18_49_21
    这个是我的标题_2020_01_01_18_47_58
  • 原文地址:https://www.cnblogs.com/zhoutao825638/p/10382024.html
Copyright © 2011-2022 走看看