19

膨胀我的布局时,我得到了这个异常:

E AndroidRuntime: android.view.InflateException: Binary XML file line #11: Binary XML file line #11: Error inflating class <unknown>
E AndroidRuntime:        at android.view.LayoutInflater.inflate(LayoutInflater.java:539)
E AndroidRuntime:        at android.view.LayoutInflater.inflate(LayoutInflater.java:423)
E AndroidRuntime:        at com.myapp.view.MyRecyclerAdapter.onCreateViewHolder(MyRecyclerAdapter:80)
E AndroidRuntime:        at android.support.v7.widget.RecyclerView$Adapter.createViewHolder(RecyclerView.java:5288)
E AndroidRuntime:        at android.support.v7.widget.RecyclerView$Recycler.getViewForPosition(RecyclerView.java:4551)
E AndroidRuntime:        at android.support.v7.widget.RecyclerView$Recycler.getViewForPosition(RecyclerView.java:4461)
E AndroidRuntime:        at android.support.v7.widget.LinearLayoutManager$LayoutState.next(LinearLayoutManager.java:1962)

日志中没有“Caused by”,但是我添加了代码来捕获异常并调用getCause(),直到它返回null,这是事件的顺序:

android.view.InflateException: Binary XML file line #11: Binary XML file line #11: Error inflating class <unknown>
android.view.InflateException: Binary XML file line #11: Error inflating class <unknown>
Error: Binary XML file line #11: Error inflating class <unknown>
android.content.res.Resources$NotFoundException: File res/drawable-v11/selectable_list_background.xml from drawable resource ID #0x7f020096
java.lang.UnsupportedOperationException: Failed to resolve attribute at index 0: TypedValue{t=0x2/d=0x7f0100f9 a=-1}

0x7f020096 是selectable_list_background(见下文),它引用的 TypedValue 是?selectableItemBackground.

我将例外范围缩小到我对?selectableItemBackground. 更具体地说,我使用的是 CardView:

<android.support.v7.widget.CardView
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:foreground="@drawable/selectable_list_background"
    card_view:cardCornerRadius="4dp"
    card_view:cardElevation="4dp"
    >

这是相关部分drawable/selectable_list_background

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:drawable="?selectableItemBackground" />
</selector>

此活动使用我自己的主题,其父主题是Theme.AppCompat.Light.NoActionBar.

这段代码曾经可以工作,但几个月后我才挖出这段代码,现在它崩溃并出现异常。我的猜测是,这与将支持库升级到 23.0.1 并将目标 SDK 升级到 23 有关,但我还没有切换回 22 来验证这一点。

如果我删除一行引用,一切正常?selectableItemBackground。我也试过?attr/selectableItemBackgroundand ?android:attr/selectableItemBackground,但得到了相同的结果。(后者让我相信支持库可能不是问题)。

编辑:

我在调试器中查看了它,我怀疑它是在android.content.res.Resources里面的这段代码loadDrawable()

Drawable loadDrawable(TypedValue value, int id, Theme theme) throws NotFoundException {
    [...]
    dr = loadDrawableForCookie(value, id, null);

注意这个函数带了一个主题,但是在调用的时候没有传入loadDrawableForCookie(),这个方法最终触发了第一个异常:

java.lang.UnsupportedOperationException: Failed to resolve attribute at index 0: TypedValue{t=0x2/d=0x7f0100f9 a=-1}
    at android.content.res.TypedArray.getDrawable(TypedArray.java:867)
    at android.graphics.drawable.StateListDrawable.inflateChildElements(StateListDrawable.java:170)
    at android.graphics.drawable.StateListDrawable.inflate(StateListDrawable.java:115)
    at android.graphics.drawable.Drawable.createFromXmlInner(Drawable.java:1215)
    at android.graphics.drawable.Drawable.createFromXml(Drawable.java:1124)
    at android.content.res.Resources.loadDrawableForCookie(Resources.java:2630)
    at android.content.res.Resources.loadDrawable(Resources.java:2540)
    at android.content.res.TypedArray.getDrawable(TypedArray.java:870)
    at android.view.View.<init>(View.java:4280)

这段代码对 Android 6.0 来说似乎是新的——5.0 的代码有很大的不同,但是在加载 drawable 时会传入主题。并且主题是必要的 AFAICT - 有问题的属性是 的一部分AppCompat,因此需要活动的主题来解决它。

不过,这似乎是一个明显的错误,所以我不相信我在这里走在正确的轨道上。有任何想法吗?

4

4 回答 4

7

正如@Artjom 所说,问题在于application context创建自定义视图时传递。我也面临同样的问题,我正在通过getApplicationContext()而不是activityas context。通过后Activity作为context问题解决。

希望它可能对其他人有所帮助。

于 2016-09-15T06:00:10.370 回答
6

官方答案 - 您不能将主题属性用于android:drawableandroid:src属性。根本不可能,设计使然。还有另一个探索相同问题并提供解决方案的答案:如何从可绘制对象中引用样式属性?

就我而言,我不能真正做到这一点,因为 AppCompat?selectableItemBackground引用了一个私有可绘制对象。所以我选择了代码解决方案 - 我将前景设置?selectableItemBackground为未选择卡时,并在我处于选择模式时设置为我自己的可绘制状态列表。

这是一个非常丑陋的解决方案,如此丑陋,以至于我不打算在这里分享代码,但到目前为止我能想到的最好的。如果有人有更好的解决方案,我很想听听。

于 2015-11-05T19:25:17.580 回答
1

不幸的是,其他答案不是我的情况。为我工作的方法是检查您的活动是否使用 AppCompat 主题样式:

    <style name="AppTheme" parent="Theme.AppCompat.Light.NoActionBar">
        <!-- Customize your theme here. -->
    </style>

然后在您的活动中使用 AppCompat 主题(在很多方面,但我展示了使用清单的方式):

<application
        android:name=".Abbott"
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">

    <!-- Any configuration inside -->

</application>
于 2018-06-18T10:55:46.000 回答
1

使用 androidx Fragment Testing 组件时我遇到了同样的错误。默认 launchFragmentInContainer()使用主题FragmentScenarioEmptyFragmentActivityTheme,导致错误。

简单的解决方法是使用您自己的扩展 AppCompat 的主题:

launchFragmentInContainer(theme = R.style.AppTheme)

于 2019-06-10T11:50:05.830 回答