gpt4 book ai didi

java - 通过 LiveData 观察器将项目添加到特定索引时出现 IndexOutOfBoundsException

转载 作者:行者123 更新时间:2023-11-30 12:06:19 25 4
gpt4 key购买 nike

我正在处理使用 FragmentStatePagerAdapter 作为适配器的 ViewPager fragment 。适配器的数据是来自数据库的列表对象,我正在观察 MediatorLiveData 的查询。MediatorLiveData 合并了我一天拥有的六个不同列表,并将它们变成一个供 Adapter 使用的列表。

我想将列表中的项目添加到特定索引并动态更新 UI。观察者在添加项目并且更新工作正常时通知适配器,但是当我尝试在特定索引上执行此操作时,调用 notifyDataSetChanged() 会导致 IndexOutOfBoundsError。

我使用如下列表初始化适配器:

public MealDetailsPagerAdapter(FragmentManager fm, List<Long> list, long date, MealViewModel vm) {
super(fm);
this.mealIdList = list;
this.date = date;
this.model = vm;
}

MealViewModel 与此问题无关,它用于适配器正在创建的 fragment 。列表更改由另一个 View 模型完成。

这是一个可以正常工作的代码:

public void changeItems(List<Long> list, long date) {
if(this.date != date){
this.mealIdList = list;
notifyDataSetChanged();
return;
}
mealIdList.addAll(list);
mealIdList = removeDuplicates(mealIdList);
System.out.println(Arrays.toString(mealIdList.toArray()));
notifyDataSetChanged();
}

调用它的观察者:

 @Override
public void onActivityCreated(@Nullable Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
mViewModel.getMealIdsInDay().observe(getViewLifecycleOwner(), longs -> {
myAdapter.changeItems(longs, mCurrentIndexDay);
if(isResumed){
mViewPager.setCurrentItem(myAdapter.getPageIndexForMealId(mViewModel.getMealId()));
isResumed=false;
}
updateIndicator(mViewPager);
});
}

默认情况下 isResumed 为 false,但是如果用户添加新的 Meal,isResumed 会更改为 true,并且 viewPager 的当前位置会更改为创建的 Meal 的位置。

但是,使用工作代码,由于 addAll() 的存在,创建的 Meal 的位置将始终位于适配器列表的末尾。我想将餐点添加到特定位置,但是如果我使用 mViewPager.getCurrentItem() 获取索引并将其发送到方法,如下所示:

    mealIdList.addAll(index, list);

addAll 本身可以工作,但 notifyDataSetChanged() 会导致 IndexOutOfBoundsError。

这是完整的堆栈跟踪:

    E/AndroidRuntime: FATAL EXCEPTION: main
Process: fi.seehowyoueat.shye.debug, PID: 14148
java.lang.IndexOutOfBoundsException: Index: 2, Size: 2
at java.util.ArrayList.set(ArrayList.java:453)
at androidx.fragment.app.FragmentStatePagerAdapter.destroyItem(FragmentStatePagerAdapter.java:147)
at androidx.viewpager.widget.ViewPager.populate(ViewPager.java:1212)
at androidx.viewpager.widget.ViewPager.setCurrentItemInternal(ViewPager.java:669)
at androidx.viewpager.widget.ViewPager.setCurrentItemInternal(ViewPager.java:631)
at androidx.viewpager.widget.ViewPager.dataSetChanged(ViewPager.java:1086)
at androidx.viewpager.widget.ViewPager$PagerObserver.onChanged(ViewPager.java:3097)
at androidx.viewpager.widget.PagerAdapter.notifyDataSetChanged(PagerAdapter.java:291)
at fi.octo3.shye.view.viewpagers.MealDetailsPagerAdapter.changeTheItems(MealDetailsPagerAdapter.java:87)
at fi.octo3.shye.fragments.MealDetailsFragment.lambda$onActivityCreated$0(MealDetailsFragment.java:223)
at fi.octo3.shye.fragments.-$$Lambda$MealDetailsFragment$XB4Svnx84FE6kVa5Gzle01e8F3o.onChanged(Unknown Source:4)
at androidx.lifecycle.LiveData.considerNotify(LiveData.java:131)
at androidx.lifecycle.LiveData.dispatchingValue(LiveData.java:149)
at androidx.lifecycle.LiveData.setValue(LiveData.java:307)
at androidx.lifecycle.MutableLiveData.setValue(MutableLiveData.java:50)
at fi.octo3.shye.models.viewmodel.-$$Lambda$2CouvY7DQv4NA0nk6EMoH6jUavw.onChanged(Unknown Source:4)
at androidx.lifecycle.MediatorLiveData$Source.onChanged(MediatorLiveData.java:152)
at androidx.lifecycle.LiveData.considerNotify(LiveData.java:131)
at androidx.lifecycle.LiveData.dispatchingValue(LiveData.java:149)
at androidx.lifecycle.LiveData.setValue(LiveData.java:307)
at androidx.lifecycle.LiveData$1.run(LiveData.java:91)
at android.os.Handler.handleCallback(Handler.java:873)
at android.os.Handler.dispatchMessage(Handler.java:99)
at android.os.Looper.loop(Looper.java:280)
at android.app.ActivityThread.main(ActivityThread.java:6748)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:493)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:858)

看着它,问题似乎是由 LiveData 引起的,但我不确定如何解决它。如果您想查看更新我拥有的 MediatorLiveData 的方法,请查看此处:

 private void refresh(long key){
if(groupsAndMealsMap.get(key) != null){
//remove source and then value from map
liveDataMerger.removeSource(groupsAndMealsMap.get(key));
groupsAndMealsMap.remove(key);
//add new value to map and add it as a source
groupsAndMealsMap.append(key, mealIdsInGroup);
liveDataMerger.addSource(groupsAndMealsMap.get(key), liveDataMerger::setValue);
}
}

refresh() 由 addItem() 调用,它从数据库中获取更新的膳食列表(列表为 mealIdsInGroup),liveDataMerger 由六个 LiveData> 对象组成。

任何帮助将不胜感激。谢谢!

编辑

这是 addItem() 方法,您可以看到服务执行程序在继续执行 refresh() 方法之前等待操作完成。

public void addItem(Meal meal, long mealGroupId){
ExecutorService service = Executors.newCachedThreadPool();
service.execute(() -> {
setMealId(db.mealDao().insertMealIntoGroup(meal, mealGroupId));
mealIdsInGroup = db.mealDao().loadMealsWithinGroup(mealGroupId);
});
service.shutdown();
try {
service.awaitTermination(Long.MAX_VALUE, TimeUnit.NANOSECONDS);
} catch (InterruptedException e) {
e.printStackTrace();
}

refresh(mealGroupId);
}

最佳答案

我怀疑是插入操作还没有结束导致的。

使用处理程序来延迟

       new Handler(getMainLooper()).postDelayed(new Runnable() {
@Override
public void run() {
notifyDataSetChanged();
}
}, 500);

500 毫秒仅用于测试。

关于java - 通过 LiveData 观察器将项目添加到特定索引时出现 IndexOutOfBoundsException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55647357/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com