近期在梳理项目中的 手机图片处理、文件处理、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)
关于该问题,网上讨论的帖子很多,也有很多的解释。
该异常恐怕只有小米系统开发人员才知道:在哪些具体条件下,会报出来。
而我们的问题,主要是处理文件时出现的。那么,此文章也只是针对手机文件处理时给出一些解决方案。
分析:
- 经测试,小米个别机型高低版本(Android版本,如android7.1.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);
以上方法,我们进行过测试,使用时,请大家酌情优化。
如果有问题,请留言或自行查询资料!!!