本文主要是介绍适配Android10.0+出现的问题及解决(不断迭代),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
项目适配Android9.0+出现的问题及解决
- 关于
- Java.lang.NoClassDefFoundError: Failed resolution of: Lorg/apache/http/conn/scheme/SchemeRegistry
- The user 10154 does not meet the requirements to access device identifiers.
- startForeground requires android.permission.FOREGROUND_SERVICE
- (xxxx.server)' ~ Channel is unrecoverably broken and will be disposed!
- No implementation found for int com.baidu.platform.comjni.tools.JNITools.initClass
- Android9.0明文访问
- Android10+权限的申请问题
- Android10+存储域变化
关于
项目大概用到了百度地图sdk、定位权限、前后台服务等。
主要涉及的权限有网络权限、定位权限(包括后台定位)、存储权限。
项目的build版本 3.5.3
targetSdkVersion 29
其实一些9.0+的问题也适配到了
Java.lang.NoClassDefFoundError: Failed resolution of: Lorg/apache/http/conn/scheme/SchemeRegistry
这个在9.0以上运行闪退会报上面这个错误,解决办法:
在AndroidManifest.xml文件的application标签里面加入如下:
<application...>
//加入如下<uses-library android:name="org.apache.http.legacy" android:required="false" /></application>
The user 10154 does not meet the requirements to access device identifiers.
首先看一下我的错误代码:
/*** 获取设备编号(对于GSM手机为IMEI;对于CDMA手机为MEID;不支持为null),需要权限android.permission.READ_PHONE_STATE*/@SuppressLint({"MissingPermission", "HardwareIds"})public static String getDeviceId(Context context) {TelephonyManager telephonyMgr = (TelephonyManager) context.getSystemService(Context.TELEPHONY_SERVICE);return telephonyMgr.getDeviceId() == null ? "" : telephonyMgr.getDeviceId();}
String deviceCode = DeviceUtils.getDeviceId(getContext());
这里通过代码获取硬件标识符。
解释一下:自 Android 10(API 级别 29)起,您的应用必须是设备或个人资料所有者应用,具有特殊运营商许可,或具有 READ_PRIVILEGED_PHONE_STATE 特权,才能访问不可重置的设备标识符
这个是运行在Android10及以上会出现的闪退问题,解决办法:
- 第一种,降低当前的targetSdkVersion,使其低于29(这显然和我写这篇文章的初心不符)
- 第二种,就是加一个判断啦,如下:
private String deviceCode;if (Build.VERSION.SDK_INT>28){deviceCode = Settings.System.getString(this.getContentResolver(), Settings.Secure.ANDROID_ID);}else {deviceCode = DeviceUtils.getDeviceId(getContext());}
这里是通过Android的自带ID取代这个硬件标识。。。
startForeground requires android.permission.FOREGROUND_SERVICE
在AndroidManifest.xml添加以下权限代码:
<uses-permission android:name="android.permission.FOREGROUND_SERVICE"/>
(xxxx.server)’ ~ Channel is unrecoverably broken and will be disposed!
一般报这个错误的时候会有这个xxx,这个xxx就是我们报错的类里面的某些方法错误,所以在这个类里面进行排查,比如类型的强转、8.0的通知服务等。
No implementation found for int com.baidu.platform.comjni.tools.JNITools.initClass
我一开始也是闪退(8.0没有提示这个),Android10闪退了,经过日志收集,发现问题,我出现的问题是lib中的so库和build中ndk里面的so库不一致,导致的缺失问题,我现在配置的是:
ndk {//cpu类型对应的.so库。abiFilters'armeabi-v7a','x86','armeabi'}
lib如下图:
Android9.0明文访问
我们添加配置文件,使其支持明文访问(http)
第一步新增network_security_config.xml文件修改:
<?xml version="1.0" encoding="utf-8"?>
<network-security-config><base-config cleartextTrafficPermitted="true"><trust-anchors><certificates src="system"/></trust-anchors></base-config>
</network-security-config>
第二步,在AndroidManifest.xml的application中添加对应标签:
android:networkSecurityConfig="@xml/network_security_config"
Android10+权限的申请问题
这里我采用的第三方申请框架,加上一些判断。
我项目中主要涉及的就是前后台定位权限、存储权限了。
第三方权限框架是郭霖老师的permissionx
定义一组权限申请列表:
//如果设置了target > 28,需要增加这个权限,否则不会弹出"始终允许"这个选择框private static final String BACK_LOCATION_PERMISSION = "android.permission.ACCESS_BACKGROUND_LOCATION";public static final String[] ARR_NEED_PERMISSIONS = new ArrayList<String>() {{add(Manifest.permission.ACCESS_COARSE_LOCATION);add(Manifest.permission.ACCESS_FINE_LOCATION);add(Manifest.permission.WRITE_EXTERNAL_STORAGE);add(Manifest.permission.READ_EXTERNAL_STORAGE);add(Manifest.permission.CAMERA);add(Manifest.permission.CALL_PHONE);add(Manifest.permission.READ_PHONE_STATE);if (Build.VERSION.SDK_INT > 28) {add(BACK_LOCATION_PERMISSION);}}}.toArray(new String[0]);
PermissionX.init(this).permissions(ARR_NEED_PERMISSIONS).request(new RequestCallback() {@Overridepublic void onResult(boolean allGranted, List<String> grantedList, List<String> deniedList) {if (allGranted){//全部授权成功,做你想做的事情}else {ToastUtils.show(R.string.permission_need);}}});
补充:
在新项目中我遇到了两个问题:
同时申请前后台定位权限有问题,需要先后申请,即先申请前台后申请后台定位权限
部分手机(Android11)获取手机信息(Manifest.permission.READ_PHONE_STATE)即使已经允许过了还是会提示被拒绝,这里我用的投巧的方法解决:
//在第三方框架被拒的方法返回里面添加判断即可,不然第一次权限申请会报拒绝了该权限,但其实去设置里面看到的是已经给了权限的if(!checkPermissionMethod(Manifest.permission.READ_PHONE_STATE)) {// 执行操作。ActivityCompat.requestPermissions(LoginActivity.this, new String[]{Manifest.permission.READ_PHONE_STATE}, 999);} else {}
/*** 判断是否需要检测,防止不停的弹框*/public boolean checkPermissionMethod(@NonNull String permission) {return ActivityCompat.checkSelfPermission(this, permission) == PackageManager.PERMISSION_GRANTED;}
Android10+存储域变化
部分手机(oppo的android11某机型)因为定制化的问题,用户无法查看data/下的文件内容,导致我们经常会使用的拍照存图片啊之类的存储会崩溃,而某些过低的版本的手机(5.0)在新的存储方法中又会崩溃,所以也需要适配:
if (Build.VERSION.SDK_INT<=Build.VERSION_CODES.M){//这个是存储在app私有目录里面的,随着app卸载一起删除savePhotoDir = this.getExternalFilesDir("xxx_Photos") +"";}else {//华为手机没有这个问题,oppo的部分手机上有,所以只能用原来的savePhotoDir = Environment.getExternalStorageDirectory() + "/xxx_Photos";}
写的稍微有点仓促,可能会有遗漏或是问题,有问题欢迎批评指正留言
这篇关于适配Android10.0+出现的问题及解决(不断迭代)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!