Android获取设备隐私 忽略6.0权限管理

一.前言
(1).由于MIUI等部分国产定制系统也有权限管理,没有相关api,故无法判断用户是否允许获取联系人等隐私。在Android 6.0之后,新增权限管理可以通过官方api判断用户的运行状态;
(2).我们指定targetSdkVersion为23或者之后我们还需要在运行时请求这些所需的权限。这很重要,因为已经出现了很多开发者把targetSdkVersion飙到了最新,然后发现自己的app疯狂的崩溃,这是由于他们没有实现执行运行时权限请求的代码。当你已经把一个targeting API 为23或者之后的app发布到了Google Play上,这更是一个问题,你无法立即把那个apk的targeting API替换成更早的版本。

二.权限分析
从Android6.0开始,权限分为普通权限和许可权限。许可权限分类归组,一个权限授权之后,该组下的权限均可使用。
(1)普通权限
只需要在xml申请即可,使用方法和之前6.0以前的一样。在应用安装应用时,会默认获得许可。
(2)许可权限
可执行 $adb shell pm list permissions -d -g

同一组的任何一个权限被授权了,其他权限也自动被授权。例如,一旦WRITE_CONTACTS被授权了,app也有READ_CONTACTS和GET_ACCOUNTS了。
源码中被用来检查和请求权限的方法分别是Activity的checkSelfPermission和requestPermissions,这些方法api23引入。

三.相关方法
(1).ContextCompat.checkSelfPermission()
检查应用是否拥有该权限,被授权返回值为PERMISSION_GRANTED,否则返回PERMISSION_DENIED
(2).ActivityCompat.requestPermissions()
将弹出请求授权对话框,这个方法在M之前版本调用,OnRequestPermissionsResultCallback 直接被调用,带着正确的 PERMISSION_GRANTED或者 PERMISSION_DENIED 。
(3).AppCompatActivity.onRequestPermissionsResult()
该方法类似于Activity的OnActivityResult()的回调方法,主要接收请求授权的返回值

//版本判断
if (Build.VERSION.SDK_INT >= 23) {
  //减少是否拥有权限
  int checkCallPhonePermission = ContextCompat.checkSelfPermission(getApplicationContext(), permission);
  if (checkCallPhonePermission != PackageManager.PERMISSION_GRANTED) {
    //弹出对话框接收权限
    ActivityCompat.requestPermissions(BaseActivity.this, new String[]{permission}, id);
    return;
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
  super.onRequestPermissionsResult(requestCode, permissions, grantResults); 

  if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
    //TODO:已授权
  } else {
    //TODO:用户拒绝
  }
}

(4).封装

public class BaseActivity extends AppCompatActivity {
  private Map<Integer, Runnable> allowablePermissionRunnables = new HashMap<>();
  private Map<Integer, Runnable> disallowablePermissionRunnables = new HashMap<>(); 

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
  } 

  /**
   * 请求权限
   * @param id 请求授权的id 唯一标识即可
   * @param permission 请求的权限
   * @param allowableRunnable 同意授权后的操作
   * @param disallowableRunnable 禁止权限后的操作
   */
  protected void requestPermission(int id, String permission, Runnable allowableRunnable, Runnable disallowableRunnable) {
    if (allowableRunnable == null) {
      throw new IllegalArgumentException("allowableRunnable == null");
    } 

    allowablePermissionRunnables.put(id, allowableRunnable);
    if (disallowableRunnable != null) {
      disallowablePermissionRunnables.put(id, disallowableRunnable);
    } 

    //版本判断
    if (Build.VERSION.SDK_INT >= 23) {
      //减少是否拥有权限
      int checkCallPhonePermission = ContextCompat.checkSelfPermission(getApplicationContext(), permission);
      if (checkCallPhonePermission != PackageManager.PERMISSION_GRANTED) {
        //弹出对话框接收权限
        ActivityCompat.requestPermissions(BaseActivity.this, new String[]{permission}, id);
        return;
      } else {
        allowableRunnable.run();
      }
    } else {
      allowableRunnable.run();
    }
  } 

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

    if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
      Runnable allowRun = allowablePermissionRunnables.get(requestCode);
      allowRun.run();
    } else {
      Runnable disallowRun = disallowablePermissionRunnables.get(requestCode);
      disallowRun.run();
    }
  }
} 
public class MainActivity extends BaseActivity implements View.OnClickListener{
  private Button btCallPhone;
  private Button btContact; 

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main); 

    btCallPhone = (Button) findViewById(R.id.call_phone);
    btContact = (Button) findViewById(R.id.contact); 

    btCallPhone.setOnClickListener(this);
    btContact.setOnClickListener(this);
  } 

  @Override
  public void onClick(View v) {
    if(v == btCallPhone){
      //拨打电话
      requestPermission(1, Manifest.permission.CALL_PHONE, new Runnable() {
        @Override
        public void run() {
          callPhone();
        }
      }, new Runnable() {
        @Override
        public void run() {
          callPhoneDenied();
        }
      });
    }else if(v == btContact){
      //读取联系人信息
      requestPermission(2, Manifest.permission.WRITE_CONTACTS, new Runnable() {
        @Override
        public void run() {
          readContact();
        }
      }, new Runnable() {
        @Override
        public void run() {
          readContactDenied();
        }
      });
    }
  } 

  private void callPhone() {
    Toast.makeText(MainActivity.this, "CALL_PHONE OK", Toast.LENGTH_SHORT)
        .show();
  } 

  private void callPhoneDenied() {
    Toast.makeText(MainActivity.this, "CALL_PHONE Denied", Toast.LENGTH_SHORT)
        .show();
  } 

  private void readContact() {
    ContentResolver cr = getContentResolver();
    String str[] = {ContactsContract.CommonDataKinds.Phone.CONTACT_ID, ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME, ContactsContract.CommonDataKinds.Phone.NUMBER,
        ContactsContract.CommonDataKinds.Phone.PHOTO_ID};
    Cursor cur = cr.query(
        ContactsContract.CommonDataKinds.Phone.CONTENT_URI, str, null,
        null, null);
    int count = cur.getCount();
    cur.close(); 

    Toast.makeText(MainActivity.this, String.format("发现%s条", count), Toast.LENGTH_SHORT)
        .show();
  } 

  private void readContactDenied() {
    Toast.makeText(MainActivity.this, "Contact Denied", Toast.LENGTH_SHORT)
        .show();
  }
}

以上就是本文的全部内容,希望对大家的学习有所帮助。

(0)

相关推荐

  • Android系统关机的全流程解析

    在PowerManager的API文档中,给出了一个关机/重启接口: public void reboot (String reason) 对于这个接口的描述很简单,就是几句话. 接口的作用就是重启设备,而且,就算重启成功了也没有返回值. 需要包含REBOOT权限,也就是android.permission.REBOOT 唯一参数reason代表需要的特定重启模式,比如recovery,当然也可以为null. 一.上层空间 1.frameworks/base/core/java/android/

  • Android 6.0开发实现关机菜单添加重启按钮的方法

    本文实例讲述了Android 6.0开发实现关机菜单添加重启按钮的方法.分享给大家供大家参考,具体如下: 修改: /trunk/LINUX/android/frameworks/base/core/res/res/values/config.xml 添加数组name为config_globalActionsList的值 修改: /LINUX/android/frameworks/base/services/core/java/com/android/server/policy/GlobalAct

  • Android开发实现长按返回键弹出关机框功能

    本文实例讲述了Android开发实现长按返回键弹出关机框功能.分享给大家供大家参考,具体如下: 今天刚好在PhoneWindowManager.java下看,当看到长按Home键的功能时,突然想到是不是可以长按back键来弹出关机框. 有想法就试试呗.当然想法是根据长按home键来的,那么我们应该可以模仿长按Home键来做.经过一番实验,貌似好像可以,拿出来给大家分享一下!!! 先找到PhoneWindowManager.java文件,在framework/base/policy/src/com

  • Android实现关机重启的方法分享

    实现系统重启的APK需要system的权限,在AndroidManifest.xml中增加android:sharedUserId="android.uid.system",再修改签名即可: 具体方法参考: 点击打开链接 1.使用PowerManager来实现:代码: 复制代码 代码如下: private void rebootSystem(){      PowerManager pManager=(PowerManager) getSystemService(Context.POW

  • Android编程实现系统重启与关机的方法

    本文实例讲述了Android编程实现系统重启与关机的方法.分享给大家供大家参考,具体如下: 最近在做个东西,巧合碰到了sharedUserId 的问题,所以收集了一些资料,存存档备份. 安装在设备中的每一个apk文件,Android 给每个 APK 进程分配一个单独的用户空间,其 manifest 中的 userid 就是对应一个 Linux 用户都会被分配到一个属于自己的统一的 Linux 用户 ID,并且为它创建一个沙箱,以防止影响其他应用程序(或者其他应用程序影响它). 用户 ID 在应用

  • Android实现关机与重启的几种方式(推荐)

    下面我们来探究Android如何实现关机,重启:在Android中这种操作往往需要管理员级别,或者root Android实现的方式如下几种: 默认的SDK并没有提供应用开发者直接的Android系统关机或重启的API接口,一般来讲,实现Android系统的关机或重启,需要较高的权限(系统权限甚至Root权限).所以,在一般的APP中,如果想要实现关机或重启功能,要么是在App中声明系统权限,要么是通过某种"间接"的方式,比如广播或反射,来间接实现系统关机或重启.再者,就是放在源码环境

  • Android手机获取root权限并实现关机重启功能的方法

    本文实例讲述了Android手机获取root权限并实现关机重启功能的方法,是Android程序设计中非常常见的重要功能.现分享给大家,供大家在Android程序开发中参考之用. 具体功能代码如下: /* * 执行命令 * @param command * 1.获取root权限 "chmod 777 "+getPackageCodePath() * 2.关机 reboot -p * 3.重启 reboot */ public static boolean execCmd(String c

  • 谈谈Android6.0运行时的权限处理

    运行时权限介绍 Android 6.0在我们原有的AndroidManifest.xml声明权限的基础上, 又新增了运行时权限动态检测,以下权限都需要在运行时判断: 1.身体传感器 2.日历    3.摄像头 4.通讯录 5.地理位置 6.麦克风 7.电话 8.短信 9.存储空间 在 Android 6.0 中,app 如果想要获得某些权限,会在应用中弹出一个对话框,让用户确认是否授予该权限. 具体的截图如下: 这要做的好处就是运行一个 app 时可以拒绝其中的某些权限,防止 app 触及到你的

  • Android6.0动态申请权限所遇到的问题小结

    白天在做SDK23版本的适配,遇到了不少坑,现在抽空记下来,以此为戒. 首先要知道哪些坑,就得先了解一些定义和基本使用方式. 那么先介绍一下动态申请的权限分组情况. 下面的权限组是由谷歌官方定义的,目的是在申请权限时,只要用户允许同一权限组的任意一条权限,那么该组的其他权限也就默认是允许的.不过据高人介绍,在使用时最好是用到哪个权限就具体的请求该权限,因为保不齐哪天谷歌一高兴就把权限组换了甚至删了 group:android.permission-group.CONTACTS permissio

  • Android6.0 屏幕固定功能详解

    可能大家看到这个标题不知道是什么东西,我先说明下,android6.0在设置->安全->屏幕固定开启后,然后再长按home键出现最近的几个Activity可以选择一个图钉按钮就开启了屏幕固定功能. 屏幕固定开启后,屏幕只能固定在设定的Task上的Activity切换. 一.设置固定屏幕 我们先来看SystemUI/src/com/android/systemui/recents/ScreenPinningRequest.java的代码,这段代码就是长按home键出现几个Activity,然后按

  • Android 关机弹出选择菜单的深入解析

    在Android系统中,长按Power键默认会弹出对话框让你选择"飞行模式","静音","关机"等功能.这些功能对于手机非常适用,但是对于机顶盒产品就没有什么必要了.本文简单介绍一下怎样定制关机界面.我的目标是长按Power键,将会关机,弹出"设备将要关机"选择对话框.如果可以选择"是"关机,和"否"返回系统.弹出对话框的代码位于:frameworks\policies\base\pho

随机推荐