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()
,或者至少让它们分别返回 1
和 0
。
通过以您的方式覆盖它们,首先您禁止ListView
的视图回收机制,其次这可能会导致您描述的崩溃,因为当您从列表中间删除一个项目时,类型删除后的视图发生了变化。
更新:实际上,这就是正在发生的事情:ListView
仅在 setAdapter()
的适配器上调用 getViewTypeCount()
并创建该大小的内部数组。因此,如果您稍后添加视图并且您的 getItemViewType()
返回大于或等于该大小的索引,您将获得 ArrayIndexOutOfBoundsException
。
更新 2:如果您需要可展开的列表视图,只需使用 ExpandableListView。
更新 3: 好的,我会尽可能详细地解释。
没有理由在屏幕外打开视图。 ListView
旨在执行回收和抑制它在 99.9% 的时间内是不需要的,并且您的情况不会落入剩余的 0.01%。
您的适配器应该记住所有细节并在 ListView 滚动时填充新出现的视图。
在您的情况下,您有两种类型的视图 - 打开和关闭,因此 getViewTypeCount()
应该返回 2 和 getItemViewType()
应该返回 0
或 1
,具体取决于视图是关闭还是打开。
在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 用户权限 )