Android权限之动态权限

  • 周涛 2016年12月16日 于安徽芜湖广电中心
  • 安卓系统版本:Android N 也就是 API24 也就是 Android 7.0 也就是 牛轧糖 [好多名字的说]
  • 请求权限:相机拍照权限

本博客内容一致同步到本人的博客站点:http://www.zhoutaotao.xyz 欢迎访问留言交流

前言

  • 安卓系统的权限管理机制从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();
            }
        } 
   }
}


本博客内容一致同步到本人的博客站点:http://www.zhoutaotao.xyz 欢迎访问留言交流

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容