Android ArrayIndexOutOfBoundsException 和 AbsListViewRecycleBin.addScrapView

Posted

技术标签:

【中文标题】Android ArrayIndexOutOfBoundsException 和 AbsListViewRecycleBin.addScrapView【英文标题】:Android ArrayIndexOutOfBoundsException and AbsListViewRecycleBin.addScrapView 【发布时间】:2014-11-03 03:53:06 【问题描述】:

我有一个随机出现的ArrayIndexOutOfBoundsException。这似乎发生在我的notifyDataSetChanged(); 期间,因为错误非常随机,很难准确指出它发生的位置。

有没有人对自定义 Adaptor 有过类似的问题?

FATAL EXCEPTION: main
 java.lang.ArrayIndexOutOfBoundsException: length=12; index=12
    at android.widget.AbsListView$RecycleBin.addScrapView(AbsListView.java:8041)
    at android.widget.ListView.layoutChildren(ListView.java:1604)
    at android.widget.AbsListView.onLayout(AbsListView.java:2444)
    at android.view.View.layout(View.java:15221)
    at android.view.ViewGroup.layout(ViewGroup.java:4793)
    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:15221)
    at android.view.ViewGroup.layout(ViewGroup.java:4793)
    at android.widget.FrameLayout.onLayout(FrameLayout.java:448)
    at android.view.View.layout(View.java:15221)
    at android.view.ViewGroup.layout(ViewGroup.java:4793)
    at android.widget.FrameLayout.onLayout(FrameLayout.java:448)
    at android.view.View.layout(View.java:15221)
    at android.view.ViewGroup.layout(ViewGroup.java:4793)
    at android.widget.FrameLayout.onLayout(FrameLayout.java:448)
    at android.view.View.layout(View.java:15221)
    at android.view.ViewGroup.layout(ViewGroup.java:4793)
    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:15221)
    at android.view.ViewGroup.layout(ViewGroup.java:4793)
    at android.widget.FrameLayout.onLayout(FrameLayout.java:448)
    at android.view.View.layout(View.java:15221)
    at android.view.ViewGroup.layout(ViewGroup.java:4793)
    at android.view.ViewRootImpl.performLayout(ViewRootImpl.java:2260)
    at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:2007)
    at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:1249)
    at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:6364)
    at android.view.Choreographer$CallbackRecord.run(Choreographer.java:791)
    at android.view.Choreographer.doCallbacks(Choreographer.java:591)
    at android.view.Choreographer.doFrame(Choreographer.java:561)
    at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:777)
    at android.os.Handler.handleCallback(Handler.java:730)
    at android.os.Handler.dispatchMessage(Handler.java:92)
    at android.os.Looper.loop(Looper.java:137)
    at android.app.ActivityThread.main(ActivityThread.java:5455)
    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:1187)
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1003)
    at dalvik.system.NativeStart.main(Native Method)




public class CarUpfitScanvinadapter extends BaseAdapter
    @SuppressWarnings("unused")
    private final String TAG = this.getClass().getSimpleName();
    private Activity mActivity;
    private ArrayList<CarUpfitModel> mData;
    private static LayoutInflater sInflater = null;
    public int height = 0;
    public int heightSet = 0;
    private CarUpfitModelForm mForm;
    private ScanlistListener mCallback;

    public CarUpfitScanvinadapter(Activity a, ArrayList<CarUpfitModel> d, Resources resLocal)
        mActivity = a;
        mData = d;
        sInflater = (LayoutInflater)mActivity.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
    

    public void removeItem(int i)
        mData.remove(i);
        this.notifyDataSetChanged();
    

    @Override
    public int getCount() 
        Log.v(TAG, "getCount");
        if(mData.size()<=0) return 1;
        return mData.size();
    

    @Override
    public boolean hasStableIds ()
        return false;
    

    @Override
    public int getViewTypeCount()                  
        return getCount();
    

    @Override
    public CarUpfitModel getItem(int position) 
        return mData.get(position);
    

    @Override
    public int getItemViewType(int position) 
        return position;
    

    @Override
    public long getItemId(int position) 
        return position;
    

    @Override
    public View getView(int position, View convertView, ViewGroup parent) 
        View vi = convertView;
        if(convertView == null)
            vi = sInflater.inflate(R.layout.subaruupfitscanlistview, parent, false);
            mForm = new CarUpfitModelForm();
            mForm.setllScanlist((LinearLayout) vi.findViewById(R.id.llScanlist));
            mForm.getllScanlist().setLayoutParams(new AbsListView.LayoutParams(LayoutParams.MATCH_PARENT,heightSet));
            mForm.setllheightset((LinearLayout) vi.findViewById(R.id.llheightset));
            mForm.setetModel((EditText) vi.findViewById(R.id.etModel));
            mForm.settvScanlistvin((TextView) vi.findViewById(R.id.tvScanlistvin));
            mForm.settvScanlistdate((TextView) vi.findViewById(R.id.tvScanlistdate));
            mForm.setcbGbkit((CheckBox) vi.findViewById(R.id.cbGbkit));
            mForm.setcbFltmat((CheckBox) vi.findViewById(R.id.cbFltmat));
            mForm.setcbFlmatbrk((CheckBox) vi.findViewById(R.id.cbFlmatbrk));
            mForm.setcbEyesight((CheckBox) vi.findViewById(R.id.cbEyesight));
            mForm.setcbTrnkpan((CheckBox) vi.findViewById(R.id.cbTrnkpan));
            mForm.setcbIntmirror((CheckBox) vi.findViewById(R.id.cbIntmirror));
            mForm.setcbExtmirror((CheckBox) vi.findViewById(R.id.cbExtmirror));
            mForm.setcbPzevbadge((CheckBox) vi.findViewById(R.id.cbPzevbadge));
            mForm.setivDelete((ImageView) vi.findViewById(R.id.ivDelete));
            mForm.setivSave((ImageView) vi.findViewById(R.id.ivSave));
            mForm.getivSave().setTag(mForm);
            vi.setTag(mForm);
        else
            mForm = (CarUpfitModelForm) vi.getTag();
        
        mForm.getllScanlist().setOnClickListener(new OnItemClickListener(position, CarUpfit.ITEM_CLICK));
        if(mData.size()<=0)
            mForm.puttvScanlistvin("No Data");
        else
            mForm.puttvScanlistvin(getItem(position).getVin());
            mForm.puttvScanlistdate(getItem(position).getDate());
            mForm.setId(getItem(position).getid());
            mForm.putjson(getItem(position).getData());
            mForm.getivSave().setOnClickListener(new OnItemClickListener(position, CarUpfit.ITEM_UPDATE));
            mForm.getivDelete().setOnClickListener(new OnItemClickListener(position, CarUpfit.ITEM_REMOVE));
        
        return vi;
    

    private class OnItemClickListener implements OnClickListener
        private int mPosition;
        private int mType;

        OnItemClickListener(int position, int type)
            mPosition = position;
            mType = type;
        
        @Override
        public void onClick(View v) 
            mCallback.scanlistclick(v, mPosition, mType);
        
    

    public void setimplements(CarUpfit sl) 
        try 
            mCallback = (ScanlistListener) sl;
         catch (ClassCastException e) 
            throw new ClassCastException("CarUpfit must implement ScanlistListener");
        
    

    public interface ScanlistListener
        public void scanlistclick(View v, int position, int type);
    

编辑

我有一个带有动画的项目列表,可以扩展列表项目。这就是我对 getViewTypeCount() 进行覆盖的原因。如果我在用户打开一个项目时不覆盖它,它也会打开其他项目。

关闭项目的屏幕截图:

项目打开的屏幕截图:

【问题讨论】:

粘贴您的适配器代码 虽然与问题无关,但你为什么要从hasStableIds()返回false @bhargavg 我试图让动画在项目上工作并能够保留他们的 ID,在解决问题时切换它。 你发布足够多的 logcat 了吗?因为我想念带有违规代码的 .java 文件。进一步length=12 index=12 应该告诉你一些原因。为什么 getViewTypeCount 等于 getCount? @greenapps 我重写了 getViewTypeCount 以便在用户单击该项目时不会打开其他项目。这是我能找到的唯一解决方案。 【参考方案1】:

如果您的所有列表项都是统一的,则不应覆盖 getViewTypeCount()getItemViewType(),或者至少让它们分别返回 10

通过以您的方式覆盖它们,首先您禁止ListView 的视图回收机制,其次这可能会导致您描述的崩溃,因为当您从列表中间删除一个项目时,类型删除后的视图发生了变化。

更新:实际上,这就是正在发生的事情:ListView 仅在 setAdapter() 的适配器上调用 getViewTypeCount() 并创建该大小的内部数组。因此,如果您稍后添加视图并且您的 getItemViewType() 返回大于或等于该大小的索引,您将获得 ArrayIndexOutOfBoundsException

更新 2:如果您需要可展开的列表视图,只需使用 ExpandableListView。

更新 3: 好的,我会尽可能详细地解释。

没有理由在屏幕外打开视图。 ListView 旨在执行回收和抑制它在 99.9% 的时间内是不需要的,并且您的情况不会落入剩余的 0.01%。

您的适配器应该记住所有细节并在 ListView 滚动时填充新出现的视图。

在您的情况下,您有两种类型的视图 - 打开和关闭,因此 getViewTypeCount() 应该返回 2 和 getItemViewType() 应该返回 01,具体取决于视图是关闭还是打开。

getView() 中,如果convertView 为空,您应该根据getItemViewType() 为该位置返回的内容来创建它。如果它不为 null,则可以保证它是您之前为同一类型创建的视图之一。

如果您的 getItemViewType() 返回任意大数(我指的是 API 19 的来源),就会发生崩溃:

// ListView.java
public void setAdapter(ListAdapter adapter) 
    ...
    // This is the only call to getViewTypeCount()
    mRecycler.setViewTypeCount(mAdapter.getViewTypeCount());
    ...


// AbsListView.java
public void setViewTypeCount(int viewTypeCount) 
    ...
    // Here it creates the array of given size
    ArrayList<View>[] scrapViews = new ArrayList[viewTypeCount];
    ...
    mViewTypeCount = viewTypeCount;
    mScrapViews = scrapViews;


void addScrapView(View scrap, int position) 
    final int viewType = lp.viewType;
    ...
    // And here you get ArrayIndexOutOfBoundsException
    mScrapViews[viewType].add(scrap);

【讨论】:

我稍后会添加视图,所以我需要告诉 getItemViewType() 添加的其他视图。错误也是随机的,有时我可以添加 20 个或更多并且没有错误。其他时候我加 2 并得到一个错误。 添加视图是什么意思?您只能添加到数组列表中。它们是另一种类型的“观点”吗?您没有说明为什么要摆弄视图类型计数,而您只有一种视图类型。 @K3NN3TH 同样,您只有一种类型的视图,因此无论您有多少视图,getViewTypeCount() 都应该返回 1。 @AntonSavin 我只有一种类型的视图,但我需要将每种视图都视为自己的视图。或者,如果我在列表中有 200 个项目并且您打开项目 1,它将打开项目 12、24 等等。因此,当用户打开项目 1 然后滚动其他项目时,其他项目将打开,这就是为什么我要覆盖它而不使用视图回收。 @K3NN3TH 这样不行,我在回答中写下了崩溃的确切原因。【参考方案2】:

当您的 getItemViewType 返回的整数超过 getViewTypeCount 的大小时也会发生这种情况,因此它们就像索引。例如。 getViewTypeCount 2 要求 getItemViewType 为 0 和 1。

【讨论】:

【参考方案3】:

您认为索引越界发生在通知数据集更改时。我同意。例如,当您删除一个项目时。该项目之后的所有项目点击侦听器将具有错误的 mPosition 值。

【讨论】:

以上是关于Android ArrayIndexOutOfBoundsException 和 AbsListViewRecycleBin.addScrapView的主要内容,如果未能解决你的问题,请参考以下文章

Android 逆向Android 权限 ( Android 逆向中使用的 android.permission 权限 | Android 系统中的 Linux 用户权限 )

android 21 是啥版本

Android逆向-Android基础逆向(2-2)

【Android笔记】android Toast

图解Android - Android核心机制

Android游戏开发大全的目录