Android:Theme.resolveAttribute返回0导致资源$ NotFoundException:资源ID#0x0

时间:2014-11-12 10:23:47

标签: android exception resources

简短问题: Resources.Theme.resolveAttibute(attrId, typVal, true)返回typVal.resourceId 0 ,导致该异常。这个ResourceReader.getResourceId()适用于整个应用的所有地方。但随机,应用程序更新或多次重启后发生此异常,因为TypedValue.resourceId为0。

长问题: 我在应用重新启动上收到 ResourcesNotFoundException ,但遗憾的是我无法确定何时发生此错误。当我在一分钟内重启(关闭,终止,重新启动,更新,设置)我的应用程序几次时,我的应用程序在第一个资源读取线上提供此RNFE。

堆栈追踪:

 Caused by: android.content.res.Resources$NotFoundException: Resource ID #0x0
        at android.content.res.Resources.getValue(Resources.java:2331)
        at android.content.res.Resources.getColor(Resources.java:2013)
        at com.my.app.navi.NavigationFragment.setUp(NavigationDrawerFragment.java:570)
        at com.my.app.main.MainActivity.onCreate(MainActivity.java:1071)

很难生成该例外,但在Google Play Developer Console中我可以看到其他设备产生的大量异常。

我查找了很多关于该异常的答案,但也没有重建,String.ValueOf(int),资源添加解决了我的问题。

我有几种不同风格和精选应用的构建风格。因为我使用attr样式引用资源。所有口味都有不同的res文件夹。主res文件夹有attrs.xml用于参考颜色,drawables和图像。所有其他代码和xml布局文件都有R.attrs引用

为了解释我的情况,这是主要的res attrs.xml

<declare-styleable name="Colors">
    <!--<<<<<<<<<<<<<<<<<<<<<  COLORS  >>>>>>>>>>>>>>>>>>>>>>>>>>>>-->
    <attr name="c_pressed_button_back" format="color"/>
    <attr name="c_window_back" format="color"/>
    <attr name="c_navigation_drawer_back" format="color"/>

flavor_one style.xml

<!--<<<<<<<<<<<<<<<<<<<<<<<<<  COLORS  >>>>>>>>>>>>>>>>>>>>>>>>>>>>>-->
    <item name="c_pressed_button_back">@color/c_theme_pressed_button_back</item>
    <item name="c_window_back">@color/c_theme_window_back</item>
    <item name="c_navigation_drawer_back">@color/c_theme_navigation_drawer_back</item>

flavor_one colors.xml

<color name="c_theme_pressed_button_back">@color/c_theme_red</color>
<color name="c_theme_window_back">@color/c_theme_green</color>
<color name="c_theme_navigation_drawer_back">@color/c_theme_blue</color>

通过这些设置,我写了这个类来进行资源阅读。

public class ResourceReader
{
    public static int getResourceId(Resources.Theme thm, int attrId)
    {
        int returnValue = 0;
        TypedValue typedValue = new TypedValue();
        try {
            thm.resolveAttribute(attrId, typedValue, true);
            returnValue = typedValue.resourceId;
        }
        catch (Exception ex) // Resource cannot resolved
        {
            ex.printStackTrace();
        }
        if(returnValue == 0)
        {
            Log.e("yucel", "Resource resolved result is 0  @thm:" + thm + "  attrid:" + attrId + "  typdata:" + typedValue.data + " typtyp:" + typedValue.type + " ");
        }
    return returnValue;
    }
}

最后我使用像

这样的资源
    int back_color_id = ResourceReader.getResourceId(appData.appContext.getTheme(), R.attr.c_navigation_drawer_back);
    int back_color=  appData.appRes.getColor(back_color_id);
    mDrawerLayout.setColor(back_color);

将抛出getColor(back_color_id)行的异常。因为back_color_id等于0.在资源阅读器thm.resolveAttribute(attrId, typedValue, true);

我认为整个代码语句看起来不错。因为整个应用程序和99%的使用这种结构运作良好。几次重启后问题发生,资源读取时随机出现问题。在那种情况下必须改变一些东西。是否有任何方法可以检查资源是否处于Context.isResourceLoading()Resoureces.isLoaded()等适当状态?

在几次重启或更新后,是否有任何一个人面临资源阅读问题?

1 个答案:

答案 0 :(得分:0)

我将attr.xml读取更改为直接主题项目阅读

    int shadow_color =  appData.appRes.getColor(R.color.c_ld_navigation_shadow_color);
    mDrawerLayout.setScrimColor(shadow_color);

这很有效。但是有一个问题。我为动态主题更改和不同的flavor变量引用创建了attr.xml引用。

在flavor_one desing中打开对话框背景,颜色名为= c_open_dialog_backcolor 在flavor_two设计中,它有一个可绘制的引用作为名为= d_open_dialog_back_pattern的xml

但是这两个设计具有相同的java代码,在设置方面没有区别,就像使用R.attr.dialog_bakground

的usign资源解析器一样

似乎{app}重新生成或更新时,Resources.Theme.resolveAttribute(int resid, TypedValue outValue, boolean resolveRefs)生成的问题无法读取资源。

我更改了我的代码并删除了所有attr.xml项目,它现在似乎很有用。感谢。