美文网首页Android开发Android技术知识Android开发经验谈
Android适配:小米手机 Unable to load re

Android适配:小米手机 Unable to load re

作者: Android技术分享 | 来源:发表于2018-03-16 12:07 被阅读0次

    近期在梳理项目中的 手机图片处理、文件处理、APK文件在线升级等功能时,发现了在个别小米手机上总是报错:
    Unable to load resource 0x00000000 from pkg=com.android.systemui

    Unable to load resource 0x00000000 from pkg=com.android.systemui
    android.content.res.Resources$NotFoundException: Resource ID #0x0
    at android.content.res.ResourcesImpl.getValue(ResourcesImpl.java:201)
    at android.content.res.MiuiResourcesImpl.getValue(MiuiResourcesImpl.java:94)
    at android.content.res.Resources.getDrawable(Resources.java:788)
    at android.graphics.drawable.Icon.loadDrawableInner(Icon.java:316)
    at android.graphics.drawable.Icon.loadDrawable(Icon.java:272)
    at android.graphics.drawable.Icon.loadDrawableAsUser(Icon.java:380)
    at com.android.systemui.statusbar.ExpandedIcon.getDrawable(ExpandedIcon.java:59)
    at com.android.systemui.statusbar.StatusBarIconView.getIcon(StatusBarIconView.java:179)
    at com.android.systemui.statusbar.StatusBarIconView.setIcon(StatusBarIconView.java:138)
    at com.android.systemui.statusbar.StatusBarIconView.updateDarkMode(StatusBarIconView.java:271)
    at com.android.systemui.statusbar.phone.SimpleStatusBar.updateDarkMode(SimpleStatusBar.java:291)
    at com.android.systemui.statusbar.phone.PhoneStatusBar$19.run(PhoneStatusBar.java:3730)
    at android.os.Handler.handleCallback(Handler.java:754)
    at android.os.Handler.dispatchMessage(Handler.java:95)
    at android.os.Looper.loop(Looper.java:163)
    at android.app.ActivityThread.main(ActivityThread.java:6363)
    at java.lang.reflect.Method.invoke(Native Method)
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:904)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:794)
    

    关于该问题,网上讨论的帖子很多,也有很多的解释。

    该异常恐怕只有小米系统开发人员才知道:在哪些具体条件下,会报出来。
    而我们的问题,主要是处理文件时出现的。那么,此文章也只是针对手机文件处理时给出一些解决方案。


    分析:

    1. 经测试,小米个别机型高低版本(Android版本,如android7.1.2等)都有可能出现该问题。
    2. 问题的主要原因:使用Intent传递参数时,直接把大的数据作为了参数进行传递,如文件。
     调用系统程序,裁切指定的图片,小米手机报此异常
       // 裁切图
        public void cutimage(Uri imageUri) {
            Intent intent= new Intent("com.android.camera.action.CROP");
            intent.setDataAndType(imageUri, "image/*");
            ......
            ......
            intent.putExtra("outputFormat", "JPEG");
            intent.putExtra("noFaceDetection", true);
            intent.putExtra("return-data", true);
            startActivityForResult(intent, AppConstants.REQUEST_CODE_CROP);
        }
    
    
    调用系统程序,安装下载的apk文件时,小米手机报此异常
     /**
         * apk安装
         */
        public static void InstallApk(Context context, File apkFile){      
            Intent intent = new Intent();
            intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
            intent.setAction(android.content.Intent.ACTION_VIEW);
            intent.setDataAndType(FileProviderUtils.uriFromFile(context, apkFile),
     "application/vnd.android.package-archive");
            context.startActivity(intent);
        }
    
    setDataAndType方法传递了安装包数据时,报异常;
    

    除此之外,还有很多例子,此处不再罗列。


    原因:

    通过网上查询资料、代码实验,这个问题应该是:因为小米个别机型使用Intent传递过大的数据导致的。

    intent.setDataAndType(imageUri, "image/*");
    该方法传递的是uri,不是文件,没有问题;
    
    Intent.putExtra("return-data",true);
    该方法是指否回传数据。
    如果设置为true,当处理完毕后,结果会通过intent进行回传,如果处理的是大文件,那就要小心了!!!
    小米系统问题可能就是出在这里,return-data的方式只适用于小数据,小米手机有可能处理完的数据仍然过大,导致的异常。
    

    解决方案

    Intent传参时,不直接传递大的文件数据。

    重点1:
    那么,我们的数据应该如何传递呢? --以上面的例子为参考,我们进行分析。
    如果我们需要处理后的数据(如裁切后的图片),可以使用URI回传,而非实际的数据;
    如果我们不需要处理好的数据,直接设置不需要回传即可。
    但是这两种方式,都需要明确设置Intent.putExtra("return-data",false);

    重点2:
    我们任何方向传递文件都需要使用URI,但是使用时,要小心一个坑,即android7.0传递uri时,应注意权限安全。
    即:针对URI,我们要分版本适配,网上有很多处理方案,也可参考,我之前写的文章:
    https://www.jianshu.com/p/bec4497c2a63


    代码实现

    代码也同样以上面的例子为参考,给予解决:

    代码1:
    URI 适配封装类:

    package com.iwangzhe.app.util.android7.urifit;
    import android.app.Activity;
    import android.content.Context;
    import android.content.Intent;
    import android.net.Uri;
    import android.os.Build;
    import android.support.v4.content.FileProvider;
    import java.io.File;
    
    /**
     * 类:FileProviderUtils Uri适配帮助类
     * 从APP向外共享的文件URI时,必须使用该类进行适配,否则在7.0以上系统,会报错:FileUriExposedException(文件Uri暴露异常)
     * 作者: qxc
     * 日期:2018/2/23.
     */
    public class FileProviderUtils {
        /**
         * 从文件获得URI
         * @param context 上下文
         * @param file 文件
         * @return 文件对应的URI
         */
        public static Uri uriFromFile(Context context, File file) {
            Uri fileUri;
            //7.0以上进行适配
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                String p = context.getPackageName() + ".FileProvider";
                fileUri = FileProvider.getUriForFile(
                        context,
                        p,
                        file);
            } else {
                fileUri = Uri.fromFile(file);
            }
            return fileUri;
        }
    
        /**
         * 设置Intent的data和类型,并赋予目标程序临时的URI读写权限
         * @param context 上下文
         * @param intent 意图
         * @param type 类型
         * @param file 文件
         * @param writeAble 是否赋予可写URI的权限
         */
        public static void setIntentDataAndType(Activity context,
                                                Intent intent,
                                                String type,
                                                File file,
                                                boolean writeAble) {
            //7.0以上进行适配
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                intent.setDataAndType(uriFromFile(context, file), type);
                //临时赋予读写Uri的权限
                intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
                if (writeAble) {
                    intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
                }
            } else {
                intent.setDataAndType(Uri.fromFile(file), type);
            }
        }
    
        /**
         * 设置Intent的data和类型,并赋予目标程序临时的URI读写权限
         * @param context 上下文
         * @param intent 意图
         * @param type 类型
         * @param fileUri 文件uri
         * @param writeAble 是否赋予可写URI的权限
         */
        public static void setIntentDataAndType(Context context,
                                                Intent intent,
                                                String type,
                                                Uri fileUri,
                                                boolean writeAble) {
            //7.0以上进行适配
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                intent.setDataAndType(fileUri, type);
                //临时赋予读写Uri的权限
                intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
                if (writeAble) {
                    intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
                }
            } else {
                intent.setDataAndType(fileUri, type);
            }
        }
    }
    

    代码2:
    系统程序交互类(调用裁切、执行安装包等等)

    package com.iwangzhe.app.util.android7.systemprogram;
    import android.app.Activity;
    import android.content.Context;
    import android.content.Intent;
    import android.graphics.Bitmap;
    import android.net.Uri;
    import android.provider.MediaStore;
    import com.iwangzhe.app.util.android7.urifit.FileProviderUtils;
    import com.iwangzhe.app.util.log.collect.exception.ExceptionProxy;
    import java.io.File;
    /**
     * 类:SystemProgramUtils 系统程序适配帮助类
     * 1. 拍照
     * 2. 相册
     * 3. 裁切
     * 4. apk安装
     * 作者: qxc
     * 日期:2018/2/23.
     */
    public class SystemProgramUtils {
        public static final int REQUEST_CODE_PAIZHAO = 1;
        public static final int REQUEST_CODE_ZHAOPIAN = 2;
        public static final int REQUEST_CODE_CAIQIE = 3;
    
        /**
         * 打开相机拍照
         */
        public static void paizhao(Activity activity, File outputFile){
            Intent intent = new Intent();
            intent.setAction("android.media.action.IMAGE_CAPTURE");
            intent.addCategory("android.intent.category.DEFAULT");
            Uri uri = FileProviderUtils.uriFromFile(activity, outputFile);
            intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);
            activity.startActivityForResult(intent, REQUEST_CODE_PAIZHAO);
        }
    
        /**
         * 打开相册
         */
        public static void zhaopian(Activity activity){
            Intent intent = new Intent();
            intent.setType("image/*");
            intent.setAction("android.intent.action.PICK");
            intent.addCategory("android.intent.category.DEFAULT");
            activity.startActivityForResult(intent, REQUEST_CODE_ZHAOPIAN);
        }
    
        /**
         * 打开图片裁切
         */
        public static void Caiqie(Activity activity, Uri uri, File outputFile) {
            Intent intent = new Intent("com.android.camera.action.CROP");
            FileProviderUtils.setIntentDataAndType(activity, intent, "image/*", uri, true);
            intent.putExtra("crop", "true");
            intent.putExtra("aspectX", 1);
            intent.putExtra("aspectY", 1);
            intent.putExtra("outputX", 300);
            intent.putExtra("outputY", 300);
            //return-data为true时,直接返回bitmap,可能会很占内存,不建议,小米等个别机型会出异常!!!
            //所以适配小米等个别机型,裁切后的图片,不能直接使用data返回,应使用uri指向
            //裁切后保存的URI,不属于我们向外共享的,所以可以使用fill://类型的URI
            Uri outputUri = Uri.fromFile(outputFile);
            intent.putExtra(MediaStore.EXTRA_OUTPUT, outputUri);
            intent.putExtra("return-data", false);
            intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
            intent.putExtra("noFaceDetection", true);
            activity.startActivityForResult(intent, REQUEST_CODE_CAIQIE);
        }
    
        /**
         * apk安装
         */
        public static void InstallApk(Context context, File apkFile){
            try {
                //设置权限
                String command = "chmod 777 " + apkFile.getPath();
                Runtime runtime = Runtime.getRuntime();
                runtime.exec(command);
    
                //安装apk
                Intent intent = new Intent();
                FileProviderUtils.setIntentDataAndType(context, intent, "application/vnd.android.package-archive", FileProviderUtils.uriFromFile(context, apkFile), true);
                intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                intent.setAction(android.content.Intent.ACTION_VIEW);
                intent.putExtra("return-data", false);
                context.startActivity(intent);
            }catch (Exception ex){
                ExceptionProxy.catchException(ex);
            }
        }
    }
    
    

    裁切方法注意点:
    Uri outputUri = Uri.fromFile(outputFile);
    intent.putExtra(MediaStore.EXTRA_OUTPUT, outputUri);
    intent.putExtra("return-data", false);
    这几句代码,就是告诉裁切程序,当裁切完成后,不用直接返回数据,而是把裁切的图片输出到outputUri上,我们自己会访问该文件获得裁切后的数据。

    安装APK注意点:
    1 安装apk前,应保证对该apk有操作的权限,如不设置,个别机型手机可能报:无法解析安装包
    2 FileProviderUtils.setIntentDataAndType是我们封装的7.0URI适配的方法
    3 intent.putExtra("return-data", false);明确告诉系统,我们不需要返回处理后的数据;

    代码3:
    调用

    //获得apk文件
    File file = getApk();//注意:这是我们自己的方法,不要理解成系统的方法
    //安装apk文件
    SystemProgramUtils.InstallApk(context, file);
    

    以上方法,我们进行过测试,使用时,请大家酌情优化。
    如果有问题,请留言或自行查询资料!!!

    相关文章

      网友评论

        本文标题:Android适配:小米手机 Unable to load re

        本文链接:https://www.haomeiwen.com/subject/mpelqftx.html