长话短说:我有一个带customAdapter(称为MathAdapter)的listView。该适配器处理类型为CalculationModel的ArrayList。默认情况下,我的列表中有〜15个CalculationModels,但是,在特殊条件下,我必须删除其中的大约6个。

由于我刚接触Android开发(我来自iOS),因此我的第一个基本问题是:

通常在哪里可以操纵ListData?就我而言,一个片段初始化适配器并传递我的数据列表。在应用程序的生命周期内,我操纵此列表并调用notifiyDataSetChanged。一切都按预期工作,并且我能够一次删除或添加1个CalculationModel而没有任何问题。

删除6个CalculationModels也没有问题,也可以正常工作。当我尝试使用以下ErrorLog重新插​​入它们时,应用程序崩溃:

04-27 07:48:59.194 1766-1766/com.example.test E/AndroidRuntime: FATAL EXCEPTION: main
   Process: com.example.test, PID: 1766
   java.lang.ArrayIndexOutOfBoundsException: length=11; index=11
       at android.widget.AbsListView$RecycleBin.addScrapView(AbsListView.java:8271)
       at android.widget.ListView.layoutChildren(ListView.java:1600)
       at android.widget.AbsListView.onLayout(AbsListView.java:2528)
       at android.view.View.layout(View.java:15655)
       at android.view.ViewGroup.layout(ViewGroup.java:4856)
       at android.widget.FrameLayout.layoutChildren(FrameLayout.java:453)
       at android.widget.FrameLayout.onLayout(FrameLayout.java:388)
       at android.view.View.layout(View.java:15655)
       at android.view.ViewGroup.layout(ViewGroup.java:4856)
       at android.widget.LinearLayout.setChildFrame(LinearLayout.java:1677)
       at android.widget.LinearLayout.layoutVertical(LinearLayout.java:1531)
       at android.widget.LinearLayout.onLayout(LinearLayout.java:1440)
       at android.view.View.layout(View.java:15655)
       at android.view.ViewGroup.layout(ViewGroup.java:4856)
       at android.widget.FrameLayout.layoutChildren(FrameLayout.java:453)
       at android.widget.FrameLayout.onLayout(FrameLayout.java:388)
       at android.view.View.layout(View.java:15655)
       at android.view.ViewGroup.layout(ViewGroup.java:4856)
       at android.widget.LinearLayout.setChildFrame(LinearLayout.java:1677)
       at android.widget.LinearLayout.layoutVertical(LinearLayout.java:1531)
       at android.widget.LinearLayout.onLayout(LinearLayout.java:1440)
       at android.view.View.layout(View.java:15655)
       at android.view.ViewGroup.layout(ViewGroup.java:4856)
       at com.example.test.TouchDisableView.onLayout(TouchDisableView.java:57)
       at android.view.View.layout(View.java:15655)
       at android.view.ViewGroup.layout(ViewGroup.java:4856)
       at android.widget.FrameLayout.layoutChildren(FrameLayout.java:453)
       at android.widget.FrameLayout.onLayout(FrameLayout.java:388)
       at android.view.View.layout(View.java:15655)
       at android.view.ViewGroup.layout(ViewGroup.java:4856)
       at android.widget.FrameLayout.layoutChildren(FrameLayout.java:453)
       at android.widget.FrameLayout.onLayout(FrameLayout.java:388)
       at android.view.View.layout(View.java:15655)
       at android.view.ViewGroup.layout(ViewGroup.java:4856)
       at android.view.ViewRootImpl.performLayout(ViewRootImpl.java:2284)
       at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:2004)
       at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:1236)
       at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:6471)
       at android.view.Choreographer$CallbackRecord.run(Choreographer.java:803)
       at android.view.Choreographer.doCallbacks(Choreographer.java:603)
       at android.view.Choreographer.doFrame(Choreographer.java:573)
       at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:789)
       at android.os.Handler.handleCallback(Handler.java:733)
       at android.os.Handler.dispatchMessage(Handler.java:95)
       at android.os.Looper.loop(Looper.java:157)
       at android.app.ActivityThread.main(ActivityThread.java:5356)
       at java.lang.reflect.Method.invokeNative(Native Method)
       at java.lang.reflect.Method.invoke(Method.java:515)
       at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1265)
       at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1081)
       at dalvik.system.NativeStart.main(Native Method)


为了清楚起见,我对Java并不陌生,我发现ArrayIndexOutOfBounds是问题所在,但是我不知道发生这种情况的地方,因为我的列表和适配器中的项目数都大于11(预期为14)

码:

    List<CalculationModel> data;
    List<CalculationModel> backupData = new ArrayList<>();

@Override
public void onResume() {
    if (mathApp.isCleanPhase) {
        removeNonClearData();
    } else {
        addNonClearData();
    }

    adapter.notifyDataSetChanged();

    super.onResume();
}


    public void removeNonClearData() {
    if (backupData.size() > 0) return;

    for (int i = 0; i < data.size(); i++) {
        CalculationModel tmp = data.get(i);

        String sName = tmp.getSName();

        if (sName.equals("r") || sName.equals("wv") || sName.equals("qv") || sName.equals("tw") || sName.equals("ti") || sName.equals("h") || sName.equals("rs")) {
            backupData.add(data.remove(i));
            i--;
        }
    }

    adapter.itemCount = data.size();
}

public void addNonClearData() {
    if (backupData.size() == 0) return;

    for (int i = 0; i < backupData.size(); i++) {
        //'r' is a default parameter, which shall be in row 6
        if (backupData.get(i).getSName().equals("r")) data.add(5, backupData.remove(i));
        else data.add(backupData.remove(i));
    }

    adapter.itemCount = data.size();

    print(data.size());
}


编辑:忘了提一下,数据基本上按照我想要的方式加载,但是当我在列表中再次向上滚动时,应用程序崩溃了。所有数据都在列表中,我想在那里查看,但是向上滚动时会崩溃。

如前所述,我只是不知道此错误的来源,因为日志没有提供有关此错误的信息...

最佳答案

经过长时间的挫折和困惑(基本上是最后两天),我终于设法了解了ListViews适配器的工作原理(至少我希望如此)。

如问题所述,我有以下设置:
一个包含数据列表的片段,以及一个适配器,它在初始化时获取此数据列表。

现在,在处理数据时,适配器有两种不同的行为。

如果仅编辑列表中的一个对象(例如,我想更改位置4上的CalculationModel的参数“值”,则一切正常,并且notifyDataSetChanged()接受更改。

如果从列表中删除一个完整的对象,则notifyDataSetChanged()也可以正常工作并完成预期的工作。

但是,如果要向列表中添加一个新对象(或者在我的情况下,重新插入已删除的对象),则必须重新创建整个适配器,因为notifyDataSetChanged()不再起作用。

感谢您的帮助,如果没有评论中的输入,我可能没有做到。

08-17 20:49