本博文为子墨原创。转载请注明出处!
http://blog.csdn.net/zimo2013/article/details/50478201

1.前言

(1).因为MIUI等部分国产定制系统也有权限管理,没有相关api。故无法推断用户是否同意获取联系人等隐私。在Android 6.0之后,新增权限管理能够通过官方api推断用户的执行状态;

(2).我们指定targetSdkVersion为23或者之后我们还须要在执行时请求这些所需的权限。这非常重要。由于已经出现了非常多开发人员把targetSdkVersion飙到了最新。然后发现自己的app疯狂的崩溃,这是由于他们没有实现执行执行时权限请求的代码。当你已经把一个targeting API 为23或者之后的app公布到了Google Play上,这更是一个问题。你无法马上把那个apk的targeting API替换成更早的版本号。

2.权限分析

从Android6.0開始。权限分为普通权限和许可权限。许可权限分类归组,一个权限授权之后,该组下的权限均可使用。

(1)普通权限

仅仅须要在xml申请就可以。用法和之前6.0曾经的一样。在应用安装应用时,会默认获得许可。

(2)许可权限

可运行 $adb shell pm list permissions -d -g

Permission Group Permissions
android.permission-group.CALENDAR
  • android.permission.READ_CALENDAR
  • android.permission.WRITE_CALENDAR
android.permission-group.CAMERA
  • android.permission.CAMERA
android.permission-group.CONTACTS
  • android.permission.READ_CONTACTS
  • android.permission.WRITE_CONTACTS
  • android.permission.GET_ACCOUNTS
android.permission-group.LOCATION
  • android.permission.ACCESS_FINE_LOCATION
  • android.permission.ACCESS_COARSE_LOCATION
android.permission-group.MICROPHONE
  • android.permission.RECORD_AUDIO
android.permission-group.PHONE
  • android.permission.READ_PHONE_STATE
  • android.permission.CALL_PHONE
  • android.permission.READ_CALL_LOG
  • android.permission.WRITE_CALL_LOG
  • com.android.voicemail.permission.ADD_VOICEMAIL
  • android.permission.USE_SIP
  • android.permission.PROCESS_OUTGOING_CALLS
android.permission-group.SENSORS
  • android.permission.BODY_SENSORS
android.permission-group.SMS
  • android.permission.SEND_SMS
  • android.permission.RECEIVE_SMS
  • android.permission.READ_SMS
  • android.permission.RECEIVE_WAP_PUSH
  • android.permission.RECEIVE_MMS
  • android.permission.READ_CELL_BROADCASTS
android.permission-group.STORAGE
  • android.permission.READ_EXTERNAL_STORAGE
  • android.permission.WRITE_EXTERNAL_STORAGE

同一组的不论什么一个权限被授权了,其它权限也自己主动被授权。比如,一旦WRITE_CONTACTS被授权了,app也有READ_CONTACTS和GET_ACCOUNTS了。
源代码中被用来检查和请求权限的方法各自是Activity的checkSelfPermission和requestPermissions。这些方法api23引入。

3.相关方法

(1).ContextCompat.checkSelfPermission()

检查应用是否拥有该权限。被授权返回值为PERMISSION_GRANTED。否则返回PERMISSION_DENIED

(2).ActivityCompat.requestPermissions()

将弹出请求授权对话框,这种方法在M之前版本号调用,OnRequestPermissionsResultCallback 直接被调用,带着正确的 PERMISSION_GRANTED或者 PERMISSION_DENIED 。

(3).AppCompatActivity.onRequestPermissionsResult()

该方法类似于Activity的OnActivityResult()的回调方法,主要接收请求授权的返回值

[java] view plaincopy
  1. //版本号推断
  2. if (Build.VERSION.SDK_INT >= 23) {
  3. //降低是否拥有权限
  4. int checkCallPhonePermission = ContextCompat.checkSelfPermission(getApplicationContext(), permission);
  5. if (checkCallPhonePermission != PackageManager.PERMISSION_GRANTED) {
  6. //弹出对话框接收权限
  7. ActivityCompat.requestPermissions(BaseActivity.this, new String[]{permission}, id);
  8. return;
  9. }
[java] view plaincopy
  1. @Override
  2. public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
  3. super.onRequestPermissionsResult(requestCode, permissions, grantResults);
  4. if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
  5. //TODO:已授权
  6. } else {
  7. //TODO:用户拒绝
  8. }
  9. }

4.封装

[java] view plaincopy
  1. public class BaseActivity extends AppCompatActivity {
  2. private Map<Integer, Runnable> allowablePermissionRunnables = new HashMap<>();
  3. private Map<Integer, Runnable> disallowablePermissionRunnables = new HashMap<>();
  4. @Override
  5. protected void onCreate(Bundle savedInstanceState) {
  6. super.onCreate(savedInstanceState);
  7. }
  8. /**
  9. * 请求权限
  10. * @param id 请求授权的id 唯一标识就可以
  11. * @param permission 请求的权限
  12. * @param allowableRunnable 允许授权后的操作
  13. * @param disallowableRunnable 禁止权限后的操作
  14. */
  15. protected void requestPermission(int id, String permission, Runnable allowableRunnable, Runnable disallowableRunnable) {
  16. if (allowableRunnable == null) {
  17. throw new IllegalArgumentException("allowableRunnable == null");
  18. }
  19. allowablePermissionRunnables.put(id, allowableRunnable);
  20. if (disallowableRunnable != null) {
  21. disallowablePermissionRunnables.put(id, disallowableRunnable);
  22. }
  23. //版本号推断
  24. if (Build.VERSION.SDK_INT >= 23) {
  25. //降低是否拥有权限
  26. int checkCallPhonePermission = ContextCompat.checkSelfPermission(getApplicationContext(), permission);
  27. if (checkCallPhonePermission != PackageManager.PERMISSION_GRANTED) {
  28. //弹出对话框接收权限
  29. ActivityCompat.requestPermissions(BaseActivity.this, new String[]{permission}, id);
  30. return;
  31. } else {
  32. allowableRunnable.run();
  33. }
  34. } else {
  35. allowableRunnable.run();
  36. }
  37. }
  38. @Override
  39. public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
  40. super.onRequestPermissionsResult(requestCode, permissions, grantResults);
  41. if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
  42. Runnable allowRun = allowablePermissionRunnables.get(requestCode);
  43. allowRun.run();
  44. } else {
  45. Runnable disallowRun = disallowablePermissionRunnables.get(requestCode);
  46. disallowRun.run();
  47. }
  48. }
  49. }
[java] view plaincopy
  1. public class MainActivity extends BaseActivity implements View.OnClickListener{
  2. private Button btCallPhone;
  3. private Button btContact;
  4. @Override
  5. protected void onCreate(Bundle savedInstanceState) {
  6. super.onCreate(savedInstanceState);
  7. setContentView(R.layout.activity_main);
  8. btCallPhone = (Button) findViewById(R.id.call_phone);
  9. btContact = (Button) findViewById(R.id.contact);
  10. btCallPhone.setOnClickListener(this);
  11. btContact.setOnClickListener(this);
  12. }
  13. @Override
  14. public void onClick(View v) {
  15. if(v == btCallPhone){
  16. //拨打电话
  17. requestPermission(1, Manifest.permission.CALL_PHONE, new Runnable() {
  18. @Override
  19. public void run() {
  20. callPhone();
  21. }
  22. }, new Runnable() {
  23. @Override
  24. public void run() {
  25. callPhoneDenied();
  26. }
  27. });
  28. }else if(v == btContact){
  29. //读取联系人信息
  30. requestPermission(2, Manifest.permission.WRITE_CONTACTS, new Runnable() {
  31. @Override
  32. public void run() {
  33. readContact();
  34. }
  35. }, new Runnable() {
  36. @Override
  37. public void run() {
  38. readContactDenied();
  39. }
  40. });
  41. }
  42. }
  43. private void callPhone() {
  44. Toast.makeText(MainActivity.this, "CALL_PHONE OK", Toast.LENGTH_SHORT)
  45. .show();
  46. }
  47. private void callPhoneDenied() {
  48. Toast.makeText(MainActivity.this, "CALL_PHONE Denied", Toast.LENGTH_SHORT)
  49. .show();
  50. }
  51. private void readContact() {
  52. ContentResolver cr = getContentResolver();
  53. String str[] = {ContactsContract.CommonDataKinds.Phone.CONTACT_ID, ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME, ContactsContract.CommonDataKinds.Phone.NUMBER,
  54. ContactsContract.CommonDataKinds.Phone.PHOTO_ID};
  55. Cursor cur = cr.query(
  56. ContactsContract.CommonDataKinds.Phone.CONTENT_URI, str, null,
  57. null, null);
  58. int count = cur.getCount();
  59. cur.close();
  60. Toast.makeText(MainActivity.this, String.format("发现%s条", count), Toast.LENGTH_SHORT)
  61. .show();
  62. }
  63. private void readContactDenied() {
  64. Toast.makeText(MainActivity.this, "Contact Denied", Toast.LENGTH_SHORT)
  65. .show();
  66. }
  67. }

源代码下载地址>>

转载于:https://www.cnblogs.com/zsychanpin/p/7286204.html

Android_设备隐私获取,忽略6.0权限管理相关推荐

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

    1.前言 (1).由于MIUI等部分国产定制系统也有权限管理,没有相关api,故无法判断用户是否允许获取联系人等隐私.在Android 6.0之后,新增权限管理可以通过官方api判断用户的运行状态: ...

  2. 【Android 逆向】修改 Android 系统文件 ( Android 逆向中需要经常修改的文件和目录 | 在 root 后的设备中获取 / 目录的 rw 权限后注意事项 )

    文章目录 一.Android 逆向中需要经常修改的文件和目录 二.在 root 后的设备中获取 / 目录的 rw 权限后注意事项 1.不要随意执行 wipe 命令 2.不要随意执行 rm 命令 一.A ...

  3. android 7.0 自启管理,一款不错的android6.0、7.0权限管理器推荐

    一款不错的android6.0.7.0权限管理器PermissionsCheckerUtil 初始化权限管理器:构造方法 private final Context mContext; public ...

  4. android拍照所需的权限,eclipse --- Android拍照,相册选择图片以及Android6.0权限管理...

    [实例简介] eclipse --- Android拍照,相册选择图片以及Android6.0权限管理 [实例截图] [核心代码] camreainandroidm └── camreainandro ...

  5. Android 启动系统相机,相册,裁剪图片及6.0权限管理

    在日常开发中,我们经常需要用到上传图片的 功能,这个时候通常有两种做法,第一种,从相机获取,第二种,从相册获取.今天这篇博客主要讲解利用系统的Intent怎样获取? 主要内容如下 - 怎样通过相机获取 ...

  6. android 打开相册的权限,Android 启动系统相机,相册,裁剪图片及6.0权限管理

    在日常开发中,我们经常需要用到上传图片的 功能,这个时候通常有两种做法,第一种,从相机获取,第二种,从相册获取.今天这篇博客主要讲解利用系统的Intent怎样获取? 主要内容如下 怎样通过相机获取我们 ...

  7. android关闭权限管理,Android6.0权限管理以及使用权限该注意的地方

    Android 6.0 Marshmallow首次增加了执行时权限管理,这对用户来说,能够更好的了解.控 制 app 涉及到的权限.然而对开发人员来说却是一件比較蛋疼的事情.须要兼容适配,并保证程序功 ...

  8. Android6.0权限管理-PermissionsDispatcher

    请查看我的个人网站 新的运行时权限仅当我们设置targetSdkVersion to 23才起作用,app在6.0之前的设备依然使用旧的权限系统. 如果app的targetSdkVersion 低于 ...

  9. Android 6.0 权限管理最佳实践

    博客: Android 6.0 运行时权限管理最佳实践 github: https://github.com/yanzhenjie/AndPermission

最新文章

  1. 阿里离职员工吐槽加班太疯狂,所有的高薪都是加班加出来的!被榨干到一丝精力都不剩!婚姻不保!...
  2. python scapy 函数_【python|scapy】sprintf输出时raw_string转string
  3. Maven父子工程配置文件详解
  4. hutool中的threadutil_Hutool - 好用的Java工具类库
  5. mysql每秒57000_MySQL 性能:使用 MySQL 5.7 实现每秒 50 万查询
  6. java sqlite3查询慢_java连接数据库进行查询优化跑不通谁能帮我调下通
  7. VS2017——50G超豪华IDE套餐酸爽体验!
  8. oneproxy mysql_在OneProxy的基础上实行MySQL读写分离与负载均衡
  9. 配置多个git账号_一台电脑,两个及多个git账号配置
  10. rpa机器人平台_RPA在财务领域的三大应用场景解析
  11. 使用link 链接外部样式和网站logo,减少html文档Style部分的信息量
  12. 2.这就是搜索引擎:核心技术详解 --- 网络爬虫
  13. java多线程-创建线程
  14. 315|大数据杀熟,如何才不被坑?
  15. python适合多大小孩学好拼音打字_都已经十七八岁的年龄了,突然发现自己居然不会拼音打字,现在想好好学,但是感觉无从下手。有什么方法么?...
  16. 企业微信集成外部APP
  17. [luogu] P1682 过家家 并查集
  18. 《精通Tableau商业数据分析与可视化》之目录
  19. Mysql 分组查询top n(多种方法)
  20. 分布式计算——Daytime协议的实现(TCP版)

热门文章

  1. CoreOS Linux Alpha的重大漏洞已修复
  2. apache 开启Gzip网页压缩
  3. Codevs2822 爱在心中
  4. 高档名片设计:12款专业的名片设计欣赏
  5. ubuntu12.04没有输入法。。
  6. 使用ASP.Net 3.5 的Ajax与Web服务开发实例
  7. 正交相机下实现滚轮按钮拖动,滚动滚轮缩放的功能
  8. Nginx内置变量以及日志格式变量参数详解
  9. 在项目中引入领域驱动设计的经验
  10. yum lock 解决方法