我一直在 java.lang.IllegalArgumentException:废弃或附加的视图可能无法回收。当使用Fragment与RecyclerView时,isScrap:false isAttached:true 。 我只有一个Activity在多个片段之间切换。关于活动onCreate我设置了默认片段,碰巧有一个RecyclerView实现完全像在文档中。在活动启动时,我获取java.lang.IllegalArgumentException:可能无法回收废弃或附加的视图。 isScrap:false isAttached:true 。
问题是如果我在开头加载一个空的片段容器,然后使用RecyclerView导航到片段,它可以正常工作。 此外,我不使用 android:animateLayoutChanges 或 notifyDataSetChanged(),如同here所述 目前我在Fragment的onResume()方法中设置了RecyclerView。我尝试将其转换为其他生命周期方法但不是运气。
感谢任何帮助。
由于
我只添加了相关的代码段。我认为这与生命周期有关,如果我没有在Activity的onCreate()中设置片段,那么它就有用了。当我有一个ListView而不是RecyclerView时,它工作。 我没有发布RecyclerView的代码,因为它与文档中的相同。
活动创建
public void onCreate(Bundle savedInstanceState) {
Log.d(TAG,"### onCreate ###");
super.onCreate(savedInstanceState);
setContentView(R.layout.efficientix_activity_layout);
if(savedInstanceState != null){
checkedSideMenuItemLabel = savedInstanceState.getInt("checkedSideMenuItemLabel");
}
//Init components
initActionBar(checkedSideMenuItemLabel,savedInstanceState);
initSideMenuArrayAdapter();
initSideMenu(checkedSideMenuItemLabel,savedInstanceState);
initActionBarDrawerToggle();
fragmentManager = this.getSupportFragmentManager();
if(savedInstanceState == null){
//Set default fragment upon activity creation.
FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();
DashboardFragment df = new DashboardFragment();
fragmentTransaction.replace(R.id.fragment_container,df,DashboardFragment.class.toString());
fragmentTransaction.commit();
}
}
片段onResume()
public void onResume() {
super.onResume();
if (recylerViewLayoutManager == null) {
recylerViewLayoutManager = new LinearLayoutManager(this.getActivity());
}
recylerView.setLayoutManager(remindersLayoutManager);
initRecylerViewAdapter();
recylerView.setAdapter(recylerViewAdapter);
}
片段布局
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent">
<android.support.v7.widget.RecyclerView android:id="@+id/myRecyclerView" style="@style/Main_Content_List_View"/>
</LinearLayout>
活动布局
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:efficientix="http://schemas.android.com/apk/res-auto"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent">
<!-- The actionbar toolbar -->
<include layout="@layout/actionbar_toolbar"/>
<android.support.v4.widget.DrawerLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/menu_layout"
android:layout_width="fill_parent"
android:layout_height="fill_parent">
<LinearLayout android:layout_width="fill_parent" android:layout_height="fill_parent"
android:id="@+id/fragment_container" android:orientation="vertical">
</LinearLayout>
<include layout="@layout/side_menu_layout"/>
</android.support.v4.widget.DrawerLayout>
</LinearLayout>
RecyclerView项目布局
<?xml version="1.0" encoding="utf-8"?>
<android.widget.TableLayout xmlns:android="http://schemas.android.com/apk/res/android"
style="@style/Main_Content_Table_Layout_List_Item">
<TableRow style="@style/Main_Content_Table_Row_List_Item">
<TextView android:id="@+id/description"
style="@style/Main_Content_TextView" />
<ImageView android:id="@+id/category"
style="@style/Main_Content_ImageView"/>
</TableRow>
<TableRow style="@style/Main_Content_Table_Row_List_Item">
<TextView android:id="@+id/alert_date"
style="@style/Main_Content_TextView" />
<TextView android:id="@+id/type"
style="@style/Main_Content_TextView" />
</TableRow>
</android.widget.TableLayout>
答案 0 :(得分:2)
好的,我发现问题是这样的。在我的ViewHolder中,我除了布局视图外,还有一个ORMlite实体(可能是任何不属于布局的对象)。问题是ViewHolder的equals()和hashcode()方法基于null的实体。它为null的原因是因为使用RecyclerView,您无法访问onCreateViewHolder()中的数据位置,而只能访问onBindViewHolder()。在我的情况下,适配器数据是ORMlite实体的列表,我想将实体捆绑在持有者中。 (还是要找到办法做到这一点......)
在这种情况下我期待NullPointerException。我通过调用holder.setIsRecyclable(true)来设法获得NullPointerException。
希望这有助于其他有需要的人。
由于
答案 1 :(得分:0)
当我在ViewHolder中为Recycler View适配器使用自定义对象时,我发现这种情况发生了。
要解决此问题,我清除了自定义对象,在我的情况下是适配器的onViewRecycled(ViewHolder持有者)中的计时器,如下所示:
public void onViewRecycled(ViewHolder holder) {
if(holder instanceof EntityViewHolder) {
if(((EntityViewHolder)holder).timer != null) {
((EntityViewHolder) holder).timer.cancel();
}
}
super.onViewRecycled(holder);
}
这解决了这个问题。