适配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及以上会出现的闪退问题,解决办法:

  1. 第一种,降低当前的targetSdkVersion,使其低于29(这显然和我写这篇文章的初心不符)
  2. 第二种,就是加一个判断啦,如下:
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如下图:
适配Android10.0+出现的问题及解决(不断迭代)_第1张图片

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() {
     
                                @Override
                                public 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 {
     
                          
                        }

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";
        }

               写的稍微有点仓促,可能会有遗漏或是问题,有问题欢迎批评指正留言

你可能感兴趣的:(常见问题专区,android)