Android获取设备隐私 忽略6.0权限管理
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-group.CAMERA
|
|
android.permission-group.CONTACTS
|
|
android.permission-group.LOCATION
|
|
android.permission-group.MICROPHONE
|
|
android.permission-group.PHONE
|
|
android.permission-group.SENSORS
|
|
android.permission-group.SMS
|
|
android.permission-group.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()的回调方法,主要接收请求授权的返回值
1
2
3
4
5
6
7
8
9
10
|
//版本判断
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 ;
}
}
|
1
2
3
4
5
6
7
8
9
|
@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.封装
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
|
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();
}
}
|
转载于:https://www.cnblogs.com/Free-Thinker/p/5801113.html
Android获取设备隐私 忽略6.0权限管理相关推荐
- Android_设备隐私获取,忽略6.0权限管理
本博文为子墨原创.转载请注明出处! http://blog.csdn.net/zimo2013/article/details/50478201 1.前言 (1).因为MIUI等部分国产定制系统也有权 ...
- android设备id完美解决方法,安卓获取渠道名渠道id Android获取设备唯一标识的终极解决方法,防止安卓7.0时崩溃问题...
一,先说获取渠道名(这里以友盟为例) /* * 4.5.1新加渠道名字段,用来传给后台去统计各个渠道下载量 * */ public static String getSource() { //获取渠道 ...
- Android检测是否有悬浮窗,Android 获取判断是否有悬浮窗权限的方法
现在很多应用都会用到悬浮窗,很多国产rom把悬浮窗权限加入控制了,你就需要判断是否有悬浮窗权限,然后做对应操作. Android 原生有自带权限管理的,只是被隐藏了.看android源码在androi ...
- android悬浮动态权限,Android 获取判断是否有悬浮窗权限的方法
现在很多应用都会用到悬浮窗,很多国产rom把悬浮窗权限加入控制了,你就需要判断是否有悬浮窗权限,然后做对应操作. Android 原生有自带权限管理的,只是被隐藏了.看android源码在androi ...
- Android获取设备状态栏status bar高度的正确姿势
Android获取设备状态栏高度的正确姿势 正确代码方式: int height = 0;int resourceId = getApplicationContext().getResources() ...
- Android获取设备ID号
Android获取设备ID号 Android有多种方式可以获取设备ID,但是每一种总会有一点限制,为此,整合几种方式在一起,这样就万无一失了 public class DeviceIdFactory ...
- Android获取设备各项信息(设备id、ip地址、设备名称、运行商、品牌、型号、分辨率、处理器、国家码、系统语言、网络类型、oaid、android版本、操作系统版本、mac地址、应用程序签名..)
Android获取设备各项信息(设备id.ip地址.APP应用名称.设备名称.运行商.品牌.型号.分辨率.处理器.国家码.系统语言.网络类型.oaid.android版本.操作系统版本.mac地址.应 ...
- Android -- 获取设备硬件信息
Android – 获取设备硬件信息 通过 Build 获取指定字段信息: 信息 方法 例子 主板 Build.BOARD PAR 系统定制商 Build.BRAND HUAWEI 硬件名称 Buil ...
- Android 获取设备号
前言 最近在对接口的时候 根据后台要求生成设备唯一id 作为key android 获取设备号比较简单 这里记录一下 实现 /*** 获取设备号* @param context* @return*/p ...
最新文章
- 洛谷——P2083 找人
- LeetCode 213 House Robber II Python
- python用pywin32库来隐藏windows文件
- mybatis一套完整入门教程
- Intel收购半导体设计公司eASIC
- 【TransE模型】基于分布式表示推理
- 加密卡华为怎么模拟_华为手机NFC模拟加密的门禁卡详细教程
- pyshp读写shapefile
- 如何用计算机计算以2为底的对数,log以2为底3的对数计算器怎么写
- WPF 加载PDF文件
- CF1293A. ConneR and the A.R.C. Markland-N
- python因数之和等于数字本身,完全数,盈数,亏数到底是什么鬼?python实现给你看...
- safari浏览器找不到服务器
- Elastic:使用 Elastic Stack 来监督系统日志及指标
- 如何修改QColorDialog窗口的背景颜色和上面的字体
- matlab箭头大小固定,科学网—Matlab 粗箭头绘制 - 肖鑫的博文
- 一个额外的小问题——关于jdk和Language level的关系
- 截至2017 年 2 月全球桌面操作系统市场份额:Linux 占 2.05%...
- 吴畏嶙2019大作业
- iPhone 13系列电池信息曝光,续航稳了