Android适配:小米手机 Unable to load resource 0x00000000 from pkg=com.android.systemui

近期在梳理项目中的 手机图片处理、文件处理、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 resource 0x00000000 from pkg=com.android.systemui)