Android为什么RecyclerView适配器崩溃?

时间:2017-01-08 03:12:11

标签: android android-recyclerview android-adapter

我有一个RecyclerView列表,可以选择两个布局,默认布局和带有适配器的CardView布局。当用户尚未创建任何CardView时,将首先显示默认UI布局。用户创建并保存CardView后,布局会切换到CardView布局。布局使用“viewType”切换,在0和1之间切换。

当我使用这两种方法时,布局之间的切换工作正常:

public int getItemCount() {
    return contactList.size()>0 ? contactList.size():1;
}

public int getItemViewType(int position) {
    return contactList.size() == 0 ? 0:1;
}

然后我将以下方法添加到另一个用途,应用程序崩溃了:

public Contact getItem(int position) {
    return contactList.get(position);
}

我在这里缺少什么?

logcat的:

  

致命的例外:主要                                                                        java.lang.RuntimeException:无法启动活动   ComponentInfo {com.example.jdw.v52 / com.wimso.v052.MainActivity}:   java.lang.IndexOutOfBoundsException:索引0无效,大小为0                                                                            在   android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2059)                                                                            在   android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2084)                                                                            在android.app.ActivityThread.access $ 600(ActivityThread.java:130)                                                                            在   android.app.ActivityThread $ H.handleMessage(ActivityThread.java:1195)                                                                            在android.os.Handler.dispatchMessage(Handler.java:99)                                                                            在android.os.Looper.loop(Looper.java:137)                                                                            在android.app.ActivityThread.main(ActivityThread.java:4745)                                                                            at java.lang.reflect.Method.invokeNative(Native Method)                                                                            在java.lang.reflect.Method.invoke(Method.java:511)                                                                            在   com.android.internal.os.ZygoteInit $ MethodAndArgsCaller.run(ZygoteInit.java:786)                                                                            在com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553)                                                                            at dalvik.system.NativeStart.main(Native Method)                                                                         引起:java.lang.IndexOutOfBoundsException:索引0,大小无效   是0                                                                            在   java.util.ArrayList.throwIndexOutOfBoundsException(ArrayList.java:251)                                                                            at java.util.ArrayList.get(ArrayList.java:304)                                                                            在   com.wimso.v052.adapter.ContactListAdapter.getItem(ContactListAdapter.java:95)                                                                            在   com.wimso.v052.adapter.ContactListAdapter.clear(ContactListAdapter.java:72)                                                                            在com.wimso.v052.MainActivity.loadData(MainActivity.java:171)                                                                            在com.wimso.v052.MainActivity.onStart(MainActivity.java:126)

Adapter.java
...
public ContactListAdapter(Context context) {
    this.context = context;
    this.contactList = new ArrayList<>();
    mLayoutInflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
} 

public void add(Contact item) {
    if (contactList.size()==0) {
        // if list is empty 
        // remove empty cards first
        contactList.clear();
        notifyDataSetChanged();
    }
    contactList.add(item);
    notifyItemInserted(contactList.size() -1);
}

public void clear() {
    while (getItemCount() > 0) {
        remove(getItem(0));
    }
}

// Remove an item from the RecyclerView/
public void remove(Contact item) {
    if (contactList.size()==0) {
        // if no more contacts in list,
        // we rebuild from scratch
        contactList.clear();
        notifyDataSetChanged();
    }
    int position = contactList.indexOf(item);
    if (position > -1) {
        contactList.remove(position);
        notifyItemRemoved(position);
        notifyItemRangeChanged(position, contactList.size());  // I added this.
    }
}

// Get the Item's position.
public Contact getItem(int position) {
    return contactList.get(position);
}

// Update the existing List of RecyclerView items.
public void addAll(List<Contact> contactList) {
    for (Contact contact : contactList) {
        add(contact);
    }
}

@Override
public int getItemCount() {
    return contactList.size()>0 ? contactList.size():1;
}

// if there are zero CardViews, use viewType 0 to get default_layout,
// otherwise provide a viewType of 1 to for each CardView in order to
// show the singlecard_layout.
@Override
public int getItemViewType(int position) {
    return contactList.size() == 0 ? 0:1;
}

// Get the Item's Id.
public long getItemId(int position) {
    return contactList.get(position).getId();
}

private static class DefaultViewHolder extends RecyclerView.ViewHolder {
    DefaultViewHolder(View itemView) {
        super(itemView);
    }
}

private class ContactViewHolder extends RecyclerView.ViewHolder {

    CountDownTimer timer;
    ...
    private ContactViewHolder(View itemView) {
        super(itemView);

    ...
    @Override
public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {

    if (viewType == 0) {
        View itemView = mLayoutInflater.inflate(R.layout.defaultcard_layout, parent, false);
        return new DefaultViewHolder(itemView);
    } else {
        View itemView = mLayoutInflater.inflate(R.layout.list_contact_item, parent, false);
...
public void onBindViewHolder(final RecyclerView.ViewHolder viewHolder, final int position) {
    int type = getItemViewType(position);
    if (type == 1) {
        Contact contact = contactList.get(position);
        final ContactViewHolder holder = (ContactViewHolder) viewHolder;
    ...

3 个答案:

答案 0 :(得分:1)

问题在于:

public int getItemCount() {
   return contactList.size()>0 ? contactList.size():1;
}

contactList的大小为0时,您仍然会使RecyclerView返回至少有1个视图。那个视图试图访问已经为0的contactList,因此:

java.lang.IndexOutOfBoundsException: Invalid index 0, size is 0

你可以这样做:

public int getItemCount() {
      return contactList.size() != null ? contactList.size() : 0;
}

如果你得到一个null contactList。

public int getItemCount() {
   return contactList.size();
}

如果您从未向适配器输入空contactList,只需返回列表大小即可。

答案 1 :(得分:1)

问题是您的适配器getItem()getItemId()未正确检查联系人列表是否为空。由于您的适配器报告的大小为1,即使该列表为空,也会调用这些方法,因为RecyclerView希望显示一个项目。

当适配器没有项目时,您似乎试图显示不同的视图。虽然可以单独在适配器中执行此操作,但我认为在布局中单独使用View并且只需在加载联系人列表时相应地更改可见性就会更简单,更简洁。这样你的适配器就不需要那么复杂了。

adapter.setItems(list); // or add, whatever
if (list.size() > 0) {
    recyclerView.setVisiblity(View.VISIBLE);
    emptyView.setVisiblity(View.GONE);
} else {
    emptyView.setVisiblity(View.VISIBLE);
    recyclerView.setVisiblity(View.GONE);
}

答案 2 :(得分:0)

我遇到了类似的崩溃,这是由于托管活动的主题。

<item name="android:windowIsTranslucent">true</item>

我的活动是透明的,片段里面有一个recyclerview。一切都会加载,但在轮换时我得到同样的错误。我的片段或recyclerview实现没有什么问题,这是活动的风格。删除修复了该问题。