Recyclerview没有调用任何适配器方法:onCreateViewHolder,onBindViewHolder,

60

我的RecyclerView没有调用onCreateViewHolder和onBindViewHolder方法,因此在RecyclerView中没有显示任何内容。我已经添加了日志进行调试,但是没有日志显示。这可能是什么问题?

我的适配器:

public class CommentListAdapter extends RecyclerView.Adapter<RecyclerView.ViewHolder>{
private static final int EMPTY_VIEW = 10 ;
private ArrayList<comment> mItems;
Boolean firstTime = true;
private Typeface mTf = null;
Context mContext;
public CommentListAdapter(Context context,ArrayList<comment> items){
    Log.e("Adapter", "constructor Called");
    this.mItems = items;
    mContext = context;
}
public class EmptyViewHolder extends RecyclerView.ViewHolder {
    public EmptyViewHolder(View itemView) {
        super(itemView);
    }
}
public class ViewHolder extends RecyclerView.ViewHolder{
    TextView mAuthorName;
    TextView mMessage;
    NetworkImageView mThumbnail;
    public ViewHolder(View itemView) {
        super(itemView);
        mAuthorName = (TextView)itemView.findViewById(R.id.author_name);
        mMessage = (TextView)itemView.findViewById(R.id.message);
        mThumbnail = (NetworkImageView)itemView.findViewById(R.id.author_avatar);
    }

}

public void add(comment item, int position) {
    mItems.add(position, item);
    notifyItemInserted(position);
}

public void remove(comment item) {
    int position = mItems.indexOf(item);
    mItems.remove(position);
    notifyItemRemoved(position);
}
@Override
public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType){
    Log.e("Adapter", "onCreateViewHolder Called");
    View v;
    if(firstTime){
        mTf = BBcTypeFace.getTypeFace(parent.getContext().getApplicationContext(),"font/bbc.ttf");
        firstTime = false;
    }
    if( viewType == EMPTY_VIEW){
        v = LayoutInflater.from(parent.getContext())
                .inflate(R.layout.comment_empty_row,parent,false);
        EmptyViewHolder evh = new EmptyViewHolder(v);
        return evh;
    }else {
        v = LayoutInflater.from(parent.getContext())
                .inflate(R.layout.comment_row, parent, false);
        ViewHolder vh = new ViewHolder(v);
        return vh;
    }
}
@Override
public void onBindViewHolder(RecyclerView.ViewHolder viewHolder, int position) {
    Log.e("Adapter", "onBindViewHolder Called");
    if(viewHolder instanceof ViewHolder) {
        ViewHolder holder = (ViewHolder)viewHolder;
        comment c = mItems.get(position);
        Log.e("Adapter", "Comment is\n: " + c.toString());
        final ViewHolder finalHolder = holder;
        ImageRequest request = new ImageRequest(c.author_img_link, new Response.Listener<Bitmap>() {
            @Override
            public void onResponse(Bitmap bitmap) {
                if (bitmap != null) {
                    finalHolder.mThumbnail.setImageBitmap(bitmap);
                }
            }
        }, 0, 0, null,
                new Response.ErrorListener() {
                    @Override
                    public void onErrorResponse(VolleyError volleyError) {
                        VolleyLog.e("ImageLoader", volleyError.getMessage());
                        VolleyLog.e("ImageLoader", volleyError.getStackTrace());
                    }
                });
        GetVideoInfo.getInstance(mContext.getApplicationContext()).addToRequestQueue(request);
        holder.mAuthorName.setText(c.author_name);
        holder.mMessage.setText(c.Message);
        holder.mMessage.setTypeface(mTf);
        holder.mAuthorName.setTypeface(mTf);
    }
}
@Override
public int getItemCount() {
    Log.e("Adapter", "getItemCount() Called");
    return (mItems.size() > 0 ? mItems.size() : 1);
}
@Override
public int getItemViewType(int position) {
    Log.e("Adapter", "getItemViewType() Called");
    if (mItems.size() == 0) {
        return EMPTY_VIEW;
    }
    return super.getItemViewType(position);
}}

我将使用public void add(comment item, int position){...}在RecyclerView中添加项目。

在片段中:

private RecyclerView mRecyclerView;
private CommentListAdapter mAdapter;
private LayoutManager mLayoutManager;

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
                         Bundle savedInstanceState) {
    View rootView = inflater.inflate(R.layout.fragment_item_detail, container, false);
....

    mLayoutManager = new LinearLayoutManager(getActivity());
    mAdapter = new CommentListAdapter(getActivity(),new ArrayList<comment>());
    mRecyclerView = (RecyclerView)rootView.findViewById(R.id.comment_list);
    mRecyclerView.setHasFixedSize(true);
    mRecyclerView.setLayoutManager(mLayoutManager);
    mRecyclerView.setAdapter(mAdapter);
    mRecyclerView.setItemAnimator(new DefaultItemAnimator());

    makeJsonObjectRequest(mItem.url);
    return rootView;
}

XML文件:

<ScrollView
xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:id="@+id/scrollView"
    android:background="@android:color/white"
    android:layout_alignParentLeft="true"
    android:layout_alignParentStart="true"

    android:layout_alignParentBottom="true">

    ....
        <RelativeLayout
            xmlns:android="http://schemas.android.com/apk/res/android"
            android:layout_width="match_parent"
            android:layout_height="match_parent">
.....

            <android.support.v7.widget.RecyclerView
                android:id="@+id/comment_list"
                android:layout_width="match_parent"
                android:layout_height="match_parent">

            </android.support.v7.widget.RecyclerView>
....
        </RelativeLayout>
    ....
</ScrollView>

comments_row 的 XML 文件:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    android:orientation="horizontal"
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent" android:layout_height="wrap_content"
android:gravity="right"
android:background="@android:color/darker_gray">

<LinearLayout
        android:orientation="vertical"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
    android:gravity="right">

        <TextView
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:textAppearance="?android:attr/textAppearanceMedium"
            android:id="@+id/author_name"
            android:layout_alignParentTop="true"
            android:layout_toLeftOf="@+id/author_avatar"
            android:layout_toStartOf="@+id/author_avatar"
            android:gravity="right" />

        <TextView
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:id="@+id/message"
            android:layout_below="@+id/author_name"
            android:layout_toLeftOf="@+id/author_avatar"
            android:layout_toStartOf="@+id/author_avatar"
            android:gravity="right" />
        </LinearLayout>

<com.android.volley.toolbox.NetworkImageView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:id="@+id/author_avatar"
    android:layout_alignParentTop="true"
    android:layout_alignParentRight="true"
    android:layout_alignParentEnd="true"
    android:minHeight="150dp"
    android:minWidth="150dp" /></LinearLayout>

我将使用以下代码向RecyclerView添加项目:

JsonObjectRequest jsonObjReqComment = new
             JsonObjectRequest(Request.Method.GET,urlJsonObj+"#comment", null,
                     new Response.Listener<JSONObject>() {
                         @Override
                         public void onResponse(JSONObject response) {
                             try {
                                 JSONArray res = response.getJSONArray("response");
                                 //Log.e("Comment","Count:"+response.toString());
                                 //Log.e("Comment","Count:"+res.length());
                                 for (int i = 0; i < res.length(); i++) {
                                     JSONObject thread = res.getJSONObject(i);
                                     JSONObject author_json = thread.getJSONObject("author");
                                     int dislikes = thread.getInt("dislikes");
                                     int likes = thread.getInt("likes");
                                     String Message = thread.getString("message");
                                     //get Author info
                                     String author_img_link = author_json.getJSONObject("avatar").getString("permalink");
                                     String author_name = author_json.getString("name");
                                     comment c = new comment(dislikes,likes,Message,author_img_link,author_name);
                                     //Log.e("Comment",c.toString());
                                     //commentsList.add(c);
                                     mAdapter.add(c,0);
                                 }

                             } catch (JSONException e) {
                                 Log.e("OnResponse","Error JSON");
                                 e.printStackTrace();
                             } catch (Exception e){
                                 Log.e("OnResponse","Error Exception");
                                 e.printStackTrace();
                             }


                         }
                     }
                     , new Response.ErrorListener() {
                 @Override
                 public void onErrorResponse(VolleyError error) {
                     VolleyLog.d("vOLLEY", "Error: " + error.getMessage());
                     // hide the progress dialog

                 }
             }
             ){
                 @Override
                 public Map<String, String> getHeaders(){
                     Map<String, String> headers = new HashMap<String, String>();
                     headers.put("User-agent", "Comment");
                     return headers;
                 }
             };

匹配父元素的宽度和高度?你确定要这样做吗? - Pedro Oliveira
那么即使是EMPTY View,也不会调用onCreate或onBind吗?另外,您是否调试过add是否被调用? - yigit
@yigit 是的,我也调用了add方法,并且看到了这个问题。 - SanatiSharif
如果 @ShivamVerma 的 getItemCount() 返回值大于等于1,我会更新我的问题。 - SanatiSharif
始终指定布局管理器。我总是忘记了这一点! - user3025587
显示剩余15条评论
9个回答

193

除了@SanatiSharif和@sohrab的答案之外,您还必须遵循以下强制步骤。

确保您调用setLayoutManager,类似下面这样。

recyclerView.setLayoutManager(new LinearLayoutManager(getContext()));

在将适配器设置到RecyclerView之前执行此操作,否则它将无法正常工作。 如果需要,您可以进行自定义。 此链接将为您提供LayoutManager的工作原理的一些想法。


29
这个回答值得更多的赞。我总是会忘记这个。 - Can Poyrazoğlu
6
我也一样,我经常会忘记这个,然后我会搜索自己可能做错的事情,最终我就来到这里哈哈。 - Ruben2112
3
问题是,为什么需要这个?这个答案解决了我的问题,但我不知道为什么。 - Ben
谢谢。那真的很有帮助。 - Siva Prakash
1
您也可以通过XML设置: app:layoutManager="androidx.recyclerview.widget.LinearLayoutManager" - Westy92
1
我已经回到这个答案的次数太多了 (o_-) - luca992

36

根据 @yigit 的猜测,ScrollViewRelativeLayout 的组合导致了这个问题,只需为 RecyclerView 腾出更多的空间即可。


1
请预览 XML 文件,确保回收站视图被正确显示。谢谢。 - blaffie
2
我的控件不在ScrollView中,但仍然遇到了这个问题。 - IgorGanapolsky
我把我的RecyclerView放在ConstraintLayout里面,所以它没有调用任何适配器方法。真是疯狂 :x - Saraz

19

7

如果不是我忘了设置布局管理器,情况可能会不同:

LinearLayoutManager layoutManager = new LinearLayoutManager(getActivity());
recycler.setLayoutManager(layoutManager);

希望能帮到你:)

6

希望这篇文章能对有需要的人有所帮助。尝试以下两种解决方法:

第一个解决方法: 确保您没有不必要地使用了这条线路。

recyclerView.setHasFixedSize(true);

第二种解决方案: 确保您将布局管理器设置为recyclerView
recycler.setLayoutManager(new LinearLayoutManager(this));

第三种解决方案:您的 getItemCount 返回0,因此 RecyclerView 永远不会尝试实例化视图。请确保返回值大于0。


3

我的情况是我使用了Fragment->ViewPager和Tablayout ->在viewpager的项中我使用了RecyclerView。

所以,我不是调用ViewPagerAdapter(getChildFragmentManager()),而是调用了ViewPagerAdapter(getSupportFragmentManager()),这就是为什么我的任何一个recycler adapter项都没有被调用。

因此,在片段中设置ViewPagerAdapter的正确方法是

ViewPagerAdapter(getChildFragmentManager())


1
在我的情况下,我有这个结构。
<ScrollView>
    <RelativeLayout>
        <android.support.v7.widget.RecyclerView/>

    </RelativeLayout>
</ScrollView>

我解决了问题,移除了相对路径。
<ScrollView>

        <android.support.v7.widget.RecyclerView/>

</ScrollView>

1

我遇到了类似的问题,但以上答案都没有帮助到我。

最后,我发现在创建和绑定Fragment视图时犯了一个错误,这个RecyclerView就在其中:

有问题的代码

FormPageFragment.onCreate

binding=FormPageBinding.inflate(layoutInflater)

但在onCreateView中,我做了这个:

FormPageFragment.onCreateView

return inflater.inflate(R.layout.form_page, container, false)

然后在onStart中设置适配器,代码如下:

FormPageFragment.onStart

binding.formContent.adapter=FormEntriesAdapter(page, viewModel.enteredData)
        

解决方案

但这是垃圾。所有的工作都应该在onCreateView中完成,当然需要将binding.rootView作为视图返回,否则绑定视图不会附加到布局上,并且在它们上设置任何适配器都没有效果。

onCreateView的可行版本

 binding=FormPageBinding.inflate(layoutInflater)
 return binding.root
        

所以这更多是与片段实例化、创建和绑定正确视图的问题,而不是与RecyclerView本身有关。

1
有点儿傻,但是阻止方法调用的另一种方法是将视图的可见性声明为GONE。
android:visibility="gone"

recyclerView.setVisibility(View.GONE);

任何一种方法都会阻止RecyclerView.Adapter中方法的调用。
我希望它能帮助到某些人。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接