在工作中,我遇到过很多次更新事故,大部分的都是权限原因。遇到这种问题的时候,一般只能通知用户卸载app,重新在官网下载最新版的app。建议开发者在app首页做一个检测的弹窗,如果发现目标用户属于更新有问题的版本,弹出引导用户去官网下载最新的apk安装。


target=29后读取系统存储

如果配置了targetSdk=29,Android10。访问系统存储空间,需要在manifest中添加属性,android:requestLegacyExternalStorage=“true”

8.0安装未知来源应用权限问题

8.0对非google play store上的应用安装,都默认禁止。必须在权限清单里加上

<uses-permission android:name="android.permission.REQUEST_INSTALL_PACKAGES" />

加上权限后,在Intent意图请求安装apk时会提醒用户去设置里打开“允许安装未知应用”。
如果不加改权限,在Intent调用安装apk服务时会什么都调不起来。

ps:要更好的处理上面这个权限问题,可以在安装时主动检测是否有该权限,主动引用用户设置该权限。

8.0通知栏问题

如果安装apk过程中,用到了通知栏去显示用户下载进度,也要考虑8.0通知栏的兼容性问题。
在需要弹出通知栏的使用,先调用下列方法

private static final String NOTIFICATION_CHANNEL_ID = "1";

    @TargetApi(26)
    private static void createNotificationChannel(Context context) {
        NotificationManager notificationManager = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
        NotificationChannel channel = new NotificationChannel(NOTIFICATION_CHANNEL_ID,
                "Channel1", NotificationManager.IMPORTANCE_DEFAULT);
        channel.enableLights(true); //是否在桌面icon右上角展示小红点
        channel.setLightColor(Color.GREEN); //小红点颜色
        channel.setShowBadge(true); //是否在久按桌面图标时显示此渠道的通知
        notificationManager.createNotificationChannel(channel);
    }

做判断,只在sdk26及以上版本调用

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O && !hasCreateNotificationChannel) {
            createNotificationChannel(context);
        }

然后在构建Notification对象的时候,传入 Channel_ID,如下:

new NotificationCompat.Builder(context, NOTIFICATION_CHANNEL_ID)

如果不像上面这样,在构建Notification之前注册Channel_ID,那么在8.0以下调用通知栏正常的代码,在8.0系统的手机上会调不出通知栏

取消Channel_ID的注册使用下例代码:

@TargetApi(26)
    public static void deleteNotificationChannel(Context context) {
        NotificationManager notificationManager = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
        notificationManager.deleteNotificationChannel(NOTIFICATION_CHANNEL_ID);
    }

7.0文件权限问题

7.0引入FileProvider机制,默认禁用文件外部共享。7.0以前正常使用的更新代码,可能7.0之后会报错。
具体的FileProvider的代码我就不贴了,网上很多,我只贴一下最后安装apk代码的一个判断

private void installApk() {
        Intent i = new Intent(Intent.ACTION_VIEW);
        Uri data;
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            data = FileProvider.getUriForFile(mContext, UrlConstants.PROVIDER_NAME, tempFile);
            // 给目标应用一个临时授权
            i.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
        } else {
            data = Uri.fromFile(tempFile);
        }
        i.setDataAndType(data, "application/vnd.android.package-archive");
        i.putExtra(MediaStore.EXTRA_OUTPUT, tempUri);
        i.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
        mContext.startActivity(i);
    }

就是需要把我下载好的apk文件,赋予一个临时的权限给apk install 服务使用,才可以正常的安装

6.0存储权限问题

一般更新apk都需要下载apk到sd卡,Android 6.0以后引入动态权限,存储权限必须代码申请才可以获取。就是在下载apk时,检查一下存储权限即可。
考虑到现在市场上基本上都是6.0以上的手机,所以在做下载apk代码的时候必须考虑权限问题。以前6.0手机还不普及的时候,比较容易忽视这个问题。