【问题标题】:Error inflating class fragment | fragment did not create a view膨胀类片段时出错 |片段没有创建视图
【发布时间】:2014-01-01 16:25:10
【问题描述】:

我遇到了这个问题:

android.view.InflateException: Binary XML file line #31: Error inflating class fragment Fragment com.myapp.mgm.QueryFragment 没有创建视图。

只有在平板电脑模式下才会出现此问题。如果应用程序在手机上运行,​​它可以正常工作。

我有这个layout-large xml 文件(fragments.xml):

<android.support.v4.widget.DrawerLayout
        xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools"
        android:id="@+id/drawer_layout"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:background="@color/lightgrey">
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:orientation="horizontal"
              android:layout_width="match_parent"
              android:layout_height="match_parent"
              android:background="@color/lightgrey"
              android:baselineAligned="false">

    <fragment
            android:name="com.myapp.mgm.QueryFragment"
            android:id="@+id/fragment1"
            android:layout_weight="1"
            android:layout_width="0dp"
            android:layout_height="match_parent"
            tools:layout="@layout/nu"/>

    <fragment
            android:name="com.myapp.mgm.Infodata"
            android:id="@+id/fragment2"
            android:layout_weight="2"
            android:layout_width="0dp"
            android:layout_height="match_parent"
            tools:layout="@layout/maindata"/>

</LinearLayout>
        <ListView
                android:id="@+id/list_slidermenu"
                android:layout_width="240dp"
                android:layout_height="match_parent"
                android:layout_gravity="start"
                android:choiceMode="singleChoice"
                android:divider="@color/orange"
                android:dividerHeight="1dp"
                android:listSelector="@drawable/menulist_selector"
                android:background="@color/lblack"/>
</android.support.v4.widget.DrawerLayout>

如您所见,我的布局上有两个片段和一个导航抽屉。

将所有这些放在一起的类是这样的:

public class Fragments extends ActionBarActivity implements QueryFragment.OnQuerySelectedListener {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.fragments);

        slidingMenu();

        if (findViewById(R.id.fragmentsboth) != null) {
            if (savedInstanceState != null) {
                return;
            }
            QueryFragment firstFragment = new QueryFragment();
            getSupportFragmentManager().beginTransaction().add(R.id.fragmentsboth, firstFragment).commit();
        }
    }
}

我的QueryFragment 课程是这样的:

public class QueryFragment extends ListFragment {
    public interface OnQuerySelectedListener {
        public void onItemSelected(Bundle itemSelectedData);
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        operation = Fragments.operation;
        if (operation == 0) {
            layoutToShow = inflater.inflate(R.layout.querylist, container, false);
            ...
        } else if (operation == 1) {
            layoutToShow = inflater.inflate(R.layout.querylist, container, false);
            ...
        } else if (operation == 2) {
            layoutToShow = inflater.inflate(R.layout.search, container, false);
            ...
        } else if (operation == 3) {
            layoutToShow = inflater.inflate(R.layout.nu, container, false);
            ...
        }
        return layoutToShow;
    }

}

还有我的InfoData 类:

public class Infodata extends Fragment implements OnSeekBarChangeListener, OnItemSelectedListener {

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        setHasOptionsMenu(true);

        View itemDataLayout = inflater.inflate(R.layout.maindata, container, false);
        ....

        if (savedInstanceState != null && savedInstanceState.size()>=2) {
            layoutType = savedInstanceState.getInt("layout");
            comesFromSavedState = true;
            itemData = savedInstanceState;
        }
        return itemDataLayout;
    }

    @Override
    public void onStart() {
        super.onStart();

        Bundle args = getArguments();
        if (args != null) {
            populateLayoutWithData(1, args);
        } else if(comesFromSavedState){
            populateLayoutWithData(2, itemData);
        }
    }

    @Override
    public void onSaveInstanceState(Bundle outState) {
        super.onSaveInstanceState(outState);
        //Saves stuff (data) in a bundle to restore after screen rotation
    }   
}

基本上我在左侧片段 (QueryFragment) 上有一个 listview,在用户选择其中一项(带有数据的光标)后,右侧片段 (InfoData) 将填充提取的数据这个光标。 如果我从左侧片段中选择一个 intem,并且它的数据填充在右侧片段中,因为我将数据保存在 onSaveInstanceState 上,如果我旋转屏幕,一切正常,也就是说,数据在右侧重新加载片段。

现在,我的 Navigadion Drawer 有 4 个选项。每个选项都会在左侧片段加载一个listView,然后像以前一样,如果用户选择其中一项,它的数据将加载到右侧片段。

这是在我的Fragments 类中处理此问题的代码:

private void selectItem(int position) {
    int typeOfLayout;
    mDrawerList.setItemChecked(position, true);
    mDrawerLayout.closeDrawer(mDrawerList);
    QueryFragment newFragment;
    FragmentTransaction transaction;

    ViewGroup parentView = (ViewGroup) findViewById(R.id.fragmentsboth);

    if (parentView != null) {
        typeOfLayout = R.id.fragmentsboth;
    } else {
        typeOfLayout = R.id.fragment1;
    }

    switch (position) {
        case 0:
            Intent i = new Intent(this, NutriData.class);
            i.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK | Intent.FLAG_ACTIVITY_NEW_TASK);
            startActivity(i);
            finish();
            break;
        case 1:
            newFragment = new QueryFragment();
            operation = 0;
            transaction = getSupportFragmentManager().beginTransaction();
            transaction.replace(typeOfLayout, newFragment).commit();
            break;
        case 2:
            newFragment = new QueryFragment();
            operation = 1;
            transaction = getSupportFragmentManager().beginTransaction();
            transaction.replace(typeOfLayout, newFragment).commit();
            break;
        case 3:
            newFragment = new QueryFragment();
            operation = 2;
            transaction = getSupportFragmentManager().beginTransaction();
            transaction.replace(typeOfLayout, newFragment).commit();
            break;
        case 4:
            newFragment = new QueryFragment();
            operation = 3;
            transaction = getSupportFragmentManager().beginTransaction();
            transaction.replace(typeOfLayout, newFragment).commit();
            break;
    }
}

想象一下这个情景:

我在左侧片段中有一个listView,然后选择其中一项。然后,它的数据被传递到正确的片段,把它放在正确的位置。如果我旋转屏幕,数据将被保存,并在旋转后重新加载到正确的片段上。现在,我打开导航抽屉并选择其中一个选项(1、2、3 或 4),其对应的 listView 已加载到左侧片段中。右边的片段继续显示用户选择的第一个项目的数据。现在,我按下左侧片段上的一个新项目,它的数据被加载到右侧片段中。甜的。它按我的意愿工作。问题是如果我现在旋转我的屏幕。我收到此错误:

01-01 15:51:16.356    2569-2569/com.myapp.mgm E/AndroidRuntime﹕ FATAL EXCEPTION: main
    java.lang.RuntimeException: Unable to start activity ComponentInfo{com.myapp.mgm/com.myapp.mgm.Fragments}: android.view.InflateException: Binary XML file line #31: Error inflating class fragment
            at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2211)
            at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2261)
            at android.app.ActivityThread.handleRelaunchActivity(ActivityThread.java:3740)
            at android.app.ActivityThread.access$700(ActivityThread.java:141)
            at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1262)
            at android.os.Handler.dispatchMessage(Handler.java:99)
            at android.os.Looper.loop(Looper.java:137)
            at android.app.ActivityThread.main(ActivityThread.java:5103)
            at java.lang.reflect.Method.invokeNative(Native Method)
            at java.lang.reflect.Method.invoke(Method.java:525)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:737)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553)
            at dalvik.system.NativeStart.main(Native Method)
     Caused by: android.view.InflateException: Binary XML file line #31: Error inflating class fragment
            at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:713)
            at android.view.LayoutInflater.rInflate(LayoutInflater.java:755)
            at android.view.LayoutInflater.rInflate(LayoutInflater.java:758)
            at android.view.LayoutInflater.inflate(LayoutInflater.java:492)
            at android.view.LayoutInflater.inflate(LayoutInflater.java:397)
            at android.view.LayoutInflater.inflate(LayoutInflater.java:353)
            at com.android.internal.policy.impl.PhoneWindow.setContentView(PhoneWindow.java:267)
            at android.app.Activity.setContentView(Activity.java:1895)
            at android.support.v7.app.ActionBarActivity.superSetContentView(ActionBarActivity.java:208)
            at android.support.v7.app.ActionBarActivityDelegateICS.setContentView(ActionBarActivityDelegateICS.java:111)
            at android.support.v7.app.ActionBarActivity.setContentView(ActionBarActivity.java:76)
            at com.myapp.mgm.Fragments.onCreate(Fragments.java:27)
            at android.app.Activity.performCreate(Activity.java:5133)
            at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1087)
            at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2175)
            at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2261)
            at android.app.ActivityThread.handleRelaunchActivity(ActivityThread.java:3740)
            at android.app.ActivityThread.access$700(ActivityThread.java:141)
            at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1262)
            at android.os.Handler.dispatchMessage(Handler.java:99)
            at android.os.Looper.loop(Looper.java:137)
            at android.app.ActivityThread.main(ActivityThread.java:5103)
            at java.lang.reflect.Method.invokeNative(Native Method)
            at java.lang.reflect.Method.invoke(Method.java:525)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:737)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553)
            at dalvik.system.NativeStart.main(Native Method)
     Caused by: java.lang.IllegalStateException: Fragment com.myapp.mgm.QueryFragment did not create a view.
            at android.support.v4.app.FragmentActivity.onCreateView(FragmentActivity.java:308)
            at android.view.LayoutInflater.createViewFromTag(LayoutInflater.java:685)
            at android.view.LayoutInflater.rInflate(LayoutInflater.java:755)
            at android.view.LayoutInflater.rInflate(LayoutInflater.java:758)
            at android.view.LayoutInflater.inflate(LayoutInflater.java:492)
            at android.view.LayoutInflater.inflate(LayoutInflater.java:397)
            at android.view.LayoutInflater.inflate(LayoutInflater.java:353)
            at com.android.internal.policy.impl.PhoneWindow.setContentView(PhoneWindow.java:267)
            at android.app.Activity.setContentView(Activity.java:1895)
            at android.support.v7.app.ActionBarActivity.superSetContentView(ActionBarActivity.java:208)
            at android.support.v7.app.ActionBarActivityDelegateICS.setContentView(ActionBarActivityDelegateICS.java:111)
            at android.support.v7.app.ActionBarActivity.setContentView(ActionBarActivity.java:76)
            at com.myapp.mgm.Fragments.onCreate(Fragments.java:27)
            at android.app.Activity.performCreate(Activity.java:5133)
            at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1087)
            at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2175)
            at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2261)
            at android.app.ActivityThread.handleRelaunchActivity(ActivityThread.java:3740)
            at android.app.ActivityThread.access$700(ActivityThread.java:141)
            at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1262)
            at android.os.Handler.dispatchMessage(Handler.java:99)
            at android.os.Looper.loop(Looper.java:137)
            at android.app.ActivityThread.main(ActivityThread.java:5103)
            at java.lang.reflect.Method.invokeNative(Native Method)
            at java.lang.reflect.Method.invoke(Method.java:525)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:737)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553)
            at dalvik.system.NativeStart.main(Native Method)

谁能解释一下我做错了什么?

【问题讨论】:

    标签: android android-layout listview android-fragments


    【解决方案1】:

    尝试删除

    xmlns:android="http://schemas.android.com/apk/res/android
    

    来自LinearLayout

    【讨论】:

    • 不。完全一样的问题
    【解决方案2】:

    您可以尝试将布局作为片段的容器,该容器将被新的片段替换。而不是直接将QueryFragment嵌入fragments.xml

    1) 片段.xml:

    <!-- change fragment1 to a layout -->
    <FrameLayout
        android:id="@+id/fragment1_container"
        .../>
    

    2) 片段 onCreate():

    setContentView(R.layout.fragments);
    
    // add QueryFragment to the container layout
    getSupportFragmentManager()
        .beginTransaction()
        .add(R.id.fragment1_container, new QueryFragment())
        .commit();
    

    3) 片段 selectItem():

    if (parentView != null) {
        typeOfLayout = R.id.fragmentsboth;
    } else {
        // change to the container layout id
        typeOfLayout = R.id.fragment1_container;
    }
    

    Error inflating class fragment | fragment did not create a view

    【讨论】:

      【解决方案3】:

      在 Android 中经常面临的一个挑战是可能频繁地破坏和重建 Activity。最常见的情况是用户在横向和纵向之间旋转设备时。发生这种崩溃的原因通常是设备方向更改导致 Android 框架在任何包含的 View 中拆除显示的 Activity,然后完全重建 Activity/View 层次结构。对 Activity 或 Views 的任何引用突然变得无效。

      注意:这个问题可以通过以下方式解决:-

      android:configChanges="orientation"android 属性在你的 android 清单中的所有活动中。这可以防止由方向变化引起的拆卸/重建过程

      另外,在 Fragment 的构造函数中不要忘记这样做

      public MyFragment()
          {
              setRetainInstance(true);
          }
      

      使用 true 值调用 setRetainInstance() 会导致 Android 在 Activity 的拆卸/重建周期中保留 Fragment。与片段或视图中包含的视图或其他对象引用一起保留。

      将此设置为 true,Android :-

      1. 持有对 Fragment 实例的引用
      2. 拆除旧的活动实例
      3. 创建一个新的 Activity 实例
      4. 将保留的 Fragment 实例附加到新的活动实例

      在尝试了所有这些疯狂的解决方案后,这为我节省了很多,这是唯一有效的解决方案。

      【讨论】:

      • 即使出于其他原因最终决定进行此更改,也只会将问题推得更深,从而减少崩溃发生的频率。也就是说,在其他情况下,Android 可能会决定拆除您的应用,而您还没有解决根本问题。例如,假设用户在运行您的应用程序时接到电话?他们接听电话,因此您的应用程序不再处于前台。然后也许他们会在设备上执行其他操作。当他们返回您的应用时,Android 可能需要重新创建所有内容。糟糕——你的潜伏错误发生了。现在如何调试?
      • 澄清:我正在评论android:configChanges="orientation",它抑制了旋转的拆卸。虽然如果您不想通过 Android 的布局机制切换到不同的布局,这可能很有用,但使用它来抑制崩溃只是避免了代码中的实际错误。它最终会咬你——首先找到并解决问题。先单独尝试setRetainInstance,确保旋转时不会出现异常。如果可行,则考虑是否抑制方向变化。
      猜你喜欢
      • 2011-09-19
      • 2013-05-28
      • 1970-01-01
      • 2014-05-19
      • 2023-03-05
      相关资源
      最近更新 更多