http://blog.youkuaiyun.com/yanzhenjie1003/article/details/52503533/
http://blog.youkuaiyun.com/xuwanli2012/article/details/52577199
国产手机适配方案
AndPermission是严格按照Android系统的运行时权限设计的,并最大限度上兼容了国产手机,目前发现的国产手机bug及解决方案:
- 部分中国厂商生产手机(例如小米某型号)的
Rationale功能,在第一次拒绝后,第二次申请时不会返回true,并且会回调申请失败,也就是说在第一次决绝后默认勾选了不再提示,所以建议一定使用SettingDialog:提示用户在系统设置中授权。 - 部分中国厂商生产手机(例如小米、华为某型号)在申请权限时,用户点击确定授权后,还是回调我们申请失败,这个时候其实我们是拥有权限的,所以我们可以在失败的方法中使用
AppOpsManager进行权限判断,AndPermission已经封装好了:
- 1
- 2
- 3
- 部分中国厂商生产手机(例如vivo、pppo某型号)在用户允许权限,并且回调了权限授权陈功的方法,但是实际执行代码时并没有这个权限,建议开发者在回调成功的方法中也利用
AppOpsManager判断下:
- 1
- 2
- 3
- 4
- 5
- 部分开发者反馈,在某些手机的
Setting中授权后实际,检查时还是没有权限,部分执行代码也是没有权限,这种手机真的兼容不到了,我也觉得没必要兼容了,建议直接放弃这种平台。
最后希望咱中国Android手机厂商早日修复这些问题,祝你们事业越来越成功,产品越做越好。
一、运行时权限介绍
Android 6.0在我们原有的AndroidManifest.xml声明权限的基础上,
又新增了运行时权限动态检测,以下权限都需要在运行时判断:
身体传感器
日历
摄像头
通讯录
地理位置
麦克风
电话
短信
存储空间
二、运行时权限处理
Android6.0系统默认为targetSdkVersion小于23的应用默认授予了所申请
的所有权限(只要在Mainifest里面声明的权限默认都给予了),所以如果你以前的APP设置的targetSdkVersion低于
23,在运行时
也不会崩溃,但这也只是一个临时的救急策略,用户还是可以在设置中取消授予
的权限或者安装的时候讲授予的权限取消,所以我们还是要学习下如何对Android
6.0以上系统进行运行时权限处理的;
1,Activity中的处理;
(1)声明目标SDK版本
我们需要在build.gradle中声明targetSdkVersion为>=23;
(2)检查并申请权限
我们需要在用到权限的地方,每次都检查是否APP已经拥有权限,
比如我们有一个下载功能,需要写SD卡的权限,我们在写入之前检查是
否有WRITE_EXTERNAL_STORAGE权限,没有则申请权限;
代码如下:
//判断是否有( 或者是否被拒绝) 写入SD卡权限
if(ContextCompat.checkSelfPermission(this,Manifest.permission.WRITE_EXTERNAL_STORAGE)!=
PackageManager.PERMISSION_GRANTED)
{
//没有权限则申请WRITE_EXTERNAL_STORAGE权限
ActivityCompat.requestPermissions(this, newString[] {Manifest.permission.WRITE_EXTERNAL_STORAGE},100);
}else {
//这里可以书写写入SD卡逻辑了
} 记得
Mainifest里面一定要写上要添加的权限否则会报错:
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
(3),请求权限后,系统会弹出请求权限的Dialog,如图:
用户选择允许或拒绝后,会回调onRequestPermissionsResult方法, 该方法类似于onActivityResult;
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if (requestCode == 100){
if (grantResults[0]==PackageManager.PERMISSION_GRANTED){//用户允许了权限
//这里可以书写写入SD卡逻辑了
Toast.makeText(this,"用户允许了权限",Toast.LENGTH_SHORT).show();
}else {
Toast.makeText(this,"用户拒绝了权限",Toast.LENGTH_SHORT).show();
}
}
} 注
:用户一单点击“允许”之后下次不再弹出授权框,如果拒绝了下次依然弹出申请权限的Dialog;
2,Fragment中运行时权限的特殊处理;
(1)在Fragment中申请权限,不要使用ActivityCompat.requestPermissions, 直接使用Fragment的
requestPermissions方法,否则会回调到Activity的 onRequestPermissionsResult中;
(2)如果在Fragment中嵌套Fragment,在子Fragment中使用requestPermissions方
法,onRequestPermissionsResult不会回调回来,建议使用 getParentFragment().requestPermissions方法,
这个方法会回调到父Fragment中的onRequestPermissionsResult,加入以下代码可以把回调透传到子Fragment:
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
List<Fragment> fragments = getChildFragmentManager().getFragments();
if (fragments != null) {
for (Fragment fragment : fragments) {
if (fragment != null) {//可以使用insteadof来判断具体的Fragment
fragment.onRequestPermissionsResult(requestCode,permissions,grantResults);
}
}
}
} 附录 以下权
限只需要在AndroidManifest.xml中声明即可使用:
android.permission.ACCESS_LOCATION_EXTRA_COMMANDS
android.permission.ACCESS_NETWORK_STATE
android.permission.ACCESS_NOTIFICATION_POLICY
android.permission.ACCESS_WIFI_STATE
android.permission.ACCESS_WIMAX_STATE
android.permission.BLUETOOTH
android.permission.BLUETOOTH_ADMIN
android.permission.BROADCAST_STICKY
android.permission.CHANGE_NETWORK_STATE
android.permission.CHANGE_WIFI_MULTICAST_STATEandroid.permission.CHANGE_WIFI_STATEandroid.permission.CHANGE_WIMAX_STATEandroid.permission.DISABLE_KEYGUARDandroid.permission.EXPAND_STATUS_BARandroid.permission.FLASHLIGHTandroid.permission.GET_ACCOUNTSandroid.permission.GET_PACKAGE_SIZEandroid.permission.INTERNETandroid.permission.KILL_BACKGROUND_PROCESSESandroid.permission.MODIFY_AUDIO_SETTINGSandroid.permission.NFCandroid.permission.READ_SYNC_SETTINGSandroid.permission.READ_SYNC_STATSandroid.permission.RECEIVE_BOOT_COMPLETEDandroid.permission.REORDER_TASKSandroid.permission.REQUEST_INSTALL_PACKAGESandroid.permission.SET_TIME_ZONEandroid.permission.SET_WALLPAPERandroid.permission.SET_WALLPAPER_HINTSandroid.permission.SUBSCRIBED_FEEDS_READandroid.permission.TRANSMIT_IRandroid.permission.USE_FINGERPRINTandroid.permission.VIBRATEandroid.permission.WAKE_LOCKandroid.permission.WRITE_SYNC_SETTINGScom.android.alarm.permission.SET_ALARMcom.android.launcher.permission.INSTALL_SHORTCUTcom.android.launcher.permission.UNINSTALL_SHORTCUT
又新增了运行时权限动态检测,以下权限都需要在运行时判断:
身体传感器
日历
摄像头
通讯录
地理位置
麦克风
电话
短信
存储空间
二、运行时权限处理
Android6.0系统默认为targetSdkVersion小于23的应用默认授予了所申请
的所有权限(只要在Mainifest里面声明的权限默认都给予了),所以如果你以前的APP设置的targetSdkVersion低于
23,在运行时
也不会崩溃,但这也只是一个临时的救急策略,用户还是可以在设置中取消授予
的权限或者安装的时候讲授予的权限取消,所以我们还是要学习下如何对Android
6.0以上系统进行运行时权限处理的;
1,Activity中的处理;
(1)声明目标SDK版本
我们需要在build.gradle中声明targetSdkVersion为>=23;
(2)检查并申请权限
我们需要在用到权限的地方,每次都检查是否APP已经拥有权限,
比如我们有一个下载功能,需要写SD卡的权限,我们在写入之前检查是
否有WRITE_EXTERNAL_STORAGE权限,没有则申请权限;
代码如下:
//判断是否有( 或者是否被拒绝) 写入SD卡权限
if(ContextCompat.checkSelfPermission(this,Manifest.permission.WRITE_EXTERNAL_STORAGE)!=
PackageManager.PERMISSION_GRANTED)
{
//没有权限则申请WRITE_EXTERNAL_STORAGE权限
ActivityCompat.requestPermissions(this, newString[] {Manifest.permission.WRITE_EXTERNAL_STORAGE},100);
}else {
//这里可以书写写入SD卡逻辑了
} 记得
Mainifest里面一定要写上要添加的权限否则会报错:
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
(3),请求权限后,系统会弹出请求权限的Dialog,如图:
用户选择允许或拒绝后,会回调onRequestPermissionsResult方法, 该方法类似于onActivityResult;
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if (requestCode == 100){
if (grantResults[0]==PackageManager.PERMISSION_GRANTED){//用户允许了权限
//这里可以书写写入SD卡逻辑了
Toast.makeText(this,"用户允许了权限",Toast.LENGTH_SHORT).show();
}else {
Toast.makeText(this,"用户拒绝了权限",Toast.LENGTH_SHORT).show();
}
}
} 注
:用户一单点击“允许”之后下次不再弹出授权框,如果拒绝了下次依然弹出申请权限的Dialog;
2,Fragment中运行时权限的特殊处理;
(1)在Fragment中申请权限,不要使用ActivityCompat.requestPermissions, 直接使用Fragment的
requestPermissions方法,否则会回调到Activity的 onRequestPermissionsResult中;
(2)如果在Fragment中嵌套Fragment,在子Fragment中使用requestPermissions方
法,onRequestPermissionsResult不会回调回来,建议使用 getParentFragment().requestPermissions方法,
这个方法会回调到父Fragment中的onRequestPermissionsResult,加入以下代码可以把回调透传到子Fragment:
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
List<Fragment> fragments = getChildFragmentManager().getFragments();
if (fragments != null) {
for (Fragment fragment : fragments) {
if (fragment != null) {//可以使用insteadof来判断具体的Fragment
fragment.onRequestPermissionsResult(requestCode,permissions,grantResults);
}
}
}
} 附录 以下权
限只需要在AndroidManifest.xml中声明即可使用:
android.permission.ACCESS_LOCATION_EXTRA_COMMANDS
android.permission.ACCESS_NETWORK_STATE
android.permission.ACCESS_NOTIFICATION_POLICY
android.permission.ACCESS_WIFI_STATE
android.permission.ACCESS_WIMAX_STATE
android.permission.BLUETOOTH
android.permission.BLUETOOTH_ADMIN
android.permission.BROADCAST_STICKY
android.permission.CHANGE_NETWORK_STATE
android.permission.CHANGE_WIFI_MULTICAST_STATEandroid.permission.CHANGE_WIFI_STATEandroid.permission.CHANGE_WIMAX_STATEandroid.permission.DISABLE_KEYGUARDandroid.permission.EXPAND_STATUS_BARandroid.permission.FLASHLIGHTandroid.permission.GET_ACCOUNTSandroid.permission.GET_PACKAGE_SIZEandroid.permission.INTERNETandroid.permission.KILL_BACKGROUND_PROCESSESandroid.permission.MODIFY_AUDIO_SETTINGSandroid.permission.NFCandroid.permission.READ_SYNC_SETTINGSandroid.permission.READ_SYNC_STATSandroid.permission.RECEIVE_BOOT_COMPLETEDandroid.permission.REORDER_TASKSandroid.permission.REQUEST_INSTALL_PACKAGESandroid.permission.SET_TIME_ZONEandroid.permission.SET_WALLPAPERandroid.permission.SET_WALLPAPER_HINTSandroid.permission.SUBSCRIBED_FEEDS_READandroid.permission.TRANSMIT_IRandroid.permission.USE_FINGERPRINTandroid.permission.VIBRATEandroid.permission.WAKE_LOCKandroid.permission.WRITE_SYNC_SETTINGScom.android.alarm.permission.SET_ALARMcom.android.launcher.permission.INSTALL_SHORTCUTcom.android.launcher.permission.UNINSTALL_SHORTCUT
本文详细介绍了Android 6.0中新增的运行时权限动态检测机制,包括权限处理流程、不同场景下的权限申请方式,以及针对国产手机的适配方案。
2291

被折叠的 条评论
为什么被折叠?



