项目大概用到了百度地图sdk、定位权限、前后台服务等。
主要涉及的权限有网络权限、定位权限(包括后台定位)、存储权限。
项目的build版本 3.5.3
targetSdkVersion 29
其实一些9.0+的问题也适配到了
这个在9.0以上运行闪退会报上面这个错误,解决办法:
在AndroidManifest.xml文件的application标签里面加入如下:
<application...>
//加入如下
<uses-library android:name="org.apache.http.legacy" android:required="false" />
</application>
首先看一下我的错误代码:
/**
* 获取设备编号(对于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及以上会出现的闪退问题,解决办法:
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取代这个硬件标识。。。
在AndroidManifest.xml添加以下权限代码:
<uses-permission android:name="android.permission.FOREGROUND_SERVICE"/>
一般报这个错误的时候会有这个xxx,这个xxx就是我们报错的类里面的某些方法错误,所以在这个类里面进行排查,比如类型的强转、8.0的通知服务等。
我一开始也是闪退(8.0没有提示这个),Android10闪退了,经过日志收集,发现问题,我出现的问题是lib中的so库和build中ndk里面的so库不一致,导致的缺失问题,我现在配置的是:
ndk {
//cpu类型对应的.so库。
abiFilters'armeabi-v7a','x86','armeabi'
}
我们添加配置文件,使其支持明文访问(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"
这里我采用的第三方申请框架,加上一些判断。
我项目中主要涉及的就是前后台定位权限、存储权限了。
第三方权限框架是郭霖老师的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 {
}
部分手机(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";
}
写的稍微有点仓促,可能会有遗漏或是问题,有问题欢迎批评指正留言