RecyclerView无限滚动问题

22

我正在尝试使用RecyclerView实现无限滚动,但我只能获取前10条记录,无法获取接下来的10条记录,甚至在尝试滚动到底部时也没有任何进展。

理应在滚动时获取接下来的10条记录,以此类推... 但只得到前10条记录。

这里我上传了我的JSON副本 - 但我无法从相同的url获取数据,所以我正在使用客户端的url和本地主机。

我正在按照这个教程进行操作。

这是我的完整代码,请问我错在哪里?

JSON :

{
  "names": [
    {
      "name": "Name 1"
    },
    {
      "name": "Name 2"
    },
    ....
    {
      "name": "Name 60"
    }
  ]
}

日志:

D/name -(13759): Name 1
D/name -(13759): Name 2
.......................
D/name -(13759): Name 60

以下是我的更新代码,我正在使用它来解析JSON数据

MainActivity.java: 已更新

public class MainActivity extends AppCompatActivity {

private Toolbar toolbar;

private TextView tvEmptyView;
private RecyclerView mRecyclerView;
private DataAdapter mAdapter;
private LinearLayoutManager mLayoutManager;

private ArrayList<Student> studentList;

protected Handler handler;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    setContentView(R.layout.activity_main);

    toolbar = (Toolbar) findViewById(R.id.toolbar);
    tvEmptyView = (TextView) findViewById(R.id.empty_view);
    mRecyclerView = (RecyclerView) findViewById(R.id.my_recycler_view);

    studentList = new ArrayList<Student>();

    handler = new Handler();
    if (toolbar != null) {
        setSupportActionBar(toolbar);
        getSupportActionBar().setTitle("Android Students");
    }

    loadData();

}

// load initial data
private void loadData() {       
    new Parser().execute("http://10.0.2.2/jsons/mytest.txt");               
}

class Parser extends AsyncTask<String, Void, Boolean> {

    ProgressDialog dialog;

    @Override
    protected void onPreExecute() {
        super.onPreExecute();
        dialog = new ProgressDialog(MainActivity.this);
        dialog.show();
        dialog.setCancelable(false);
    }

    @Override
    protected Boolean doInBackground(String... urls) {
        try {
            //------------------>>
            HttpGet httppost = new HttpGet(urls[0]);
            HttpClient httpclient = new DefaultHttpClient();
            HttpResponse response = httpclient.execute(httppost);

            // StatusLine stat = response.getStatusLine();
            int status = response.getStatusLine().getStatusCode();

            if (status == 200) {
                HttpEntity entity = response.getEntity();
                String data = EntityUtils.toString(entity);

                JSONObject jsono = new JSONObject(data);
                JSONArray jarray = jsono.getJSONArray("names");

                for (int i = 0; i < jarray.length(); i++) {
                    JSONObject object = jarray.getJSONObject(i);

                    Student actor = new Student();

                    actor.setName(object.getString("name"));
                    Log.d("name - ", object.getString("name"));

                    studentList.add(actor);             
                }

                Log.d("MainActivity:StudentList ", "The size "+studentList.size());                 

                return true;
            }

            //------------------>>

        } catch (ParseException e1) {
            e1.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        } catch (JSONException e) {
            e.printStackTrace();
        }
        return false;

    }

    protected void onPostExecute(Boolean result) {
        dialog.cancel();

        Log.d("MainActivity:StudentList ", "The size "+studentList.size());

        ArrayList< Student > temArray = 
                 new ArrayList< Student >(studentList.subList(0, 10));           
        mAdapter = new DataAdapter(temArray, mRecyclerView);

        Log.d("MainActivity:TempList ", "The size "+temArray.size());

        // set the adapter object to the Recyclerview
        mRecyclerView.setAdapter(mAdapter);     
        // use this setting to improve performance if you know that changes
        // in content do not change the layout size of the RecyclerView
        mRecyclerView.setHasFixedSize(true);

        mLayoutManager = new LinearLayoutManager(MainActivity.this);

        // use a linear layout manager
        mRecyclerView.setLayoutManager(mLayoutManager);

        if (studentList.isEmpty()) {
            mRecyclerView.setVisibility(View.GONE);
            tvEmptyView.setVisibility(View.VISIBLE);
        } else {
            mRecyclerView.setVisibility(View.VISIBLE);
            tvEmptyView.setVisibility(View.GONE);
        }

        mAdapter.setOnLoadMoreListener(new OnLoadMoreListener() {
            @Override
            public void onLoadMore() {
                //add null , so the adapter will check view_type and show progress bar at bottom
                studentList.add(null);
                mAdapter.notifyItemInserted(studentList.size() - 1);

                handler.postDelayed(new Runnable() {
                    @Override
                    public void run() {
                        //   remove progress item
                        studentList.remove(studentList.size() - 1);
                        mAdapter.notifyItemRemoved(studentList.size());
                        //add items one by one
                        int start = studentList.size();
                        int end = start + 10;

                        for (int i = start + 1; i < end; i++) {
                            mAdapter.notifyItemInserted(studentList.size());
                        }
                        mAdapter.setLoaded();
                       //or you can add all at once but do not forget to call mAdapter.notifyDataSetChanged();
                    }
                }, 2000);
            }
        });
    }
}

}

DataAdapter.java:

public class DataAdapter extends RecyclerView.Adapter {
    private final int VIEW_ITEM = 1;
    private final int VIEW_PROG = 0;

    private List<Student> studentList;

    // The minimum amount of items to have below your current scroll position
    // before loading more.
    private int visibleThreshold = 5;
    private int lastVisibleItem, totalItemCount;
    private boolean loading;
    private OnLoadMoreListener onLoadMoreListener;

    public DataAdapter(List<Student> students, RecyclerView recyclerView) {
        studentList = students;

        if (recyclerView.getLayoutManager() instanceof LinearLayoutManager) {

            final LinearLayoutManager linearLayoutManager = (LinearLayoutManager) recyclerView
                    .getLayoutManager();


                    recyclerView
                    .addOnScrollListener(new RecyclerView.OnScrollListener() {
                        @Override
                        public void onScrolled(RecyclerView recyclerView,
                                               int dx, int dy) {
                            super.onScrolled(recyclerView, dx, dy);

                            totalItemCount = linearLayoutManager.getItemCount();
                            lastVisibleItem = linearLayoutManager
                                    .findLastVisibleItemPosition();
                            if (!loading
                                    && totalItemCount <= (lastVisibleItem + visibleThreshold)) {
                                // End has been reached
                                // Do something
                                if (onLoadMoreListener != null) {
                                    onLoadMoreListener.onLoadMore();
                                }
                                loading = true;
                            }
                        }
                    });
        }
    }

    @Override
    public int getItemViewType(int position) {
        return studentList.get(position) != null ? VIEW_ITEM : VIEW_PROG;
    }

    @Override
    public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent,
            int viewType) {
        RecyclerView.ViewHolder vh;
        if (viewType == VIEW_ITEM) {
            View v = LayoutInflater.from(parent.getContext()).inflate(
                    R.layout.list_row, parent, false);

            vh = new StudentViewHolder(v);
        } else {
            View v = LayoutInflater.from(parent.getContext()).inflate(
                    R.layout.progress_item, parent, false);

            vh = new ProgressViewHolder(v);
        }
        return vh;
    }

    @Override
    public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
        if (holder instanceof StudentViewHolder) {

            Student singleStudent= (Student) studentList.get(position);

            ((StudentViewHolder) holder).tvName.setText(singleStudent.getName());

            ((StudentViewHolder) holder).student= singleStudent;

        } else {
            ((ProgressViewHolder) holder).progressBar.setIndeterminate(true);
        }
    }

    public void setLoaded() {
        loading = false;
    }

    @Override
    public int getItemCount() {
        return studentList.size();
    }

    public void setOnLoadMoreListener(OnLoadMoreListener onLoadMoreListener) {
        this.onLoadMoreListener = onLoadMoreListener;
    }


    //
    public static class StudentViewHolder extends RecyclerView.ViewHolder {
        public TextView tvName;     

        public Student student;

        public StudentViewHolder(View v) {
            super(v);
            tvName = (TextView) v.findViewById(R.id.tvName);

            v.setOnClickListener(new OnClickListener() {

                @Override
                public void onClick(View v) {
                    Toast.makeText(v.getContext(),
                            "OnClick :" + student.getName(),
                            Toast.LENGTH_SHORT).show();

                }
            });
        }
    }

    public static class ProgressViewHolder extends RecyclerView.ViewHolder {
        public ProgressBar progressBar;

        public ProgressViewHolder(View v) {
            super(v);
            progressBar = (ProgressBar) v.findViewById(R.id.progressBar1);
        }
    }
}

OnLoadMoreListener.java:

public interface OnLoadMoreListener {
     void onLoadMore();
}

1
你能否在LogCat中记录“data”并查看它打印了什么? - Satyen Udeshi
你不应该在loadMore函数内部循环添加项。 - subhash
由于您一次性加载了所有项目,我认为您将不需要“加载更多”。您没有在“post execute”中更改列表视图的可见性。请在异步任务的“post execute”中添加以下代码:`if (studentList.isEmpty()) { mRecyclerView.setVisibility(View.GONE); tvEmptyView.setVisibility(View.VISIBLE); } else { mRecyclerView.setVisibility(View.VISIBLE); tvEmptyView.setVisibility(View.GONE); }` - subhash
由于 for (int i = 1; i <= 10; i++) { studentList.add(new Student()); } 得到了前11个黑项,请删除这些代码行。 - ρяσѕρєя K
1
你看过 https://github.com/weixiao1984/Android-Infinite-Scroll-Listview 上的代码吗? - Jon
显示剩余2条评论
8个回答

4

我曾经遇到过同样的问题,使用以下代码解决了这个问题... 首先,创建这个类:

public abstract class EndlessOnScrollListener extends OnScrollListener {

    public static String TAG = EndlessOnScrollListener.class.getSimpleName();

    // use your LayoutManager instead
    private LinearLayoutManager llm;

    public EndlessOnScrollListener(LinearLayoutManager sglm) {
        this.lm = llm;
    }

    @Override
    public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
        super.onScrolled(recyclerView, dx, dy);

        if (!recyclerView.canScrollVertically(1)) {
            onScrolledToEnd();
        }
    }

    public abstract void onScrolledToEnd();
}

第二步,在您的活动中使用以下代码:

recyclerView.addOnScrollListener(new EndlessOnScrollListener() {

    @Override
    public void onScrolledToEnd() {
        if (!loading) {
            loading = true;
            // add 10 by 10 to tempList then notify changing in data
        }
        loading = false;
    }
});

这对我有效...希望它对你也有效。


1
尝试使用notifyItemRangeChanged。
yourCurrentList.addAll(newData);    
mAdapter.notifyItemRangeChanged(yourCurretList.size() + 1, newDataSize);

我想这会对您有所帮助。

1
替换
private List<Student> studentList;

带有。
private List<Object> list;

也替换

@Override
public int getItemViewType(int position) {
    return studentList.get(position) != null ? VIEW_ITEM : VIEW_PROG;
}

带有。
@Override
public int getItemViewType(int position) {
  return list.get(position) instanceof Student ?  VIEW_ITEM : VIEW_PROG;
}

要检测列表的结束,您可以使用

@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
            super.onScrollStateChanged(recyclerView, newState);

            // When went to the end of the list, load more posts
            if (newState == RecyclerView.SCROLL_STATE_IDLE) {

                if (linearLayoutManager.findLastVisibleItemPosition() >= linearLayoutManager.getItemCount() - 1) {

                    // Grow List
                }
            }
}

此外,对于添加加载项。在适配器中添加以下代码。
public void addLoadingView(){
   list.add(new Object());
   notifyDataSetChanged();
}

0

1.

获取前11个recyclerview项目为空(并持续显示进度条),请参见下面的屏幕截图:

loadData方法更改为:

private void loadData() {
  new Parser().execute("http://clienturl.com/jsons/mytest.txt");  
 }

2.

本来我应该获取前10条记录,然后在滚动时获取下10条记录,以此类推...

ParseronPostExecute方法更改为:

protected void onPostExecute(Boolean result) {
    dialog.cancel();
    ArrayList< Student > temArray = 
                 new ArrayList< Student >(studentList.subList(0, 10));           
    mAdapter = new DataAdapter(temArray, mRecyclerView);

    // set the adapter object to the Recyclerview
    mRecyclerView.setAdapter(mAdapter);
 }

同时,也要从onCreate方法中删除以下行:

    mAdapter = new DataAdapter(studentList, mRecyclerView);

    // set the adapter object to the Recyclerview
    mRecyclerView.setAdapter(mAdapter);

@Oreo:好的,将mAdapter.setOnLoadMoreListener方法的代码移动到mRecyclerView.setAdapter(mAdapter);行之后的onPostExecute中。 - ρяσѕρєя K
@Oreo:获取studentList中的数据吗?使用日志检查studentListtemArray ArrayList大小。 - ρяσѕρєя K
@Oreo:请在 onPostExecute 方法中添加调试器,并检查它是否正在执行? - ρяσѕρєя K
让我们在聊天中继续这个讨论 - ρяσѕρєя K
@Oreo:哎呀,不是这样的。问题还没有解决吗? - ρяσѕρєя K
显示剩余4条评论

0

尝试更改

int i = start + 1; i <= end; i++

在 for 循环中

int i = start + 1; i < end; i++

<= 验证添加了一个额外的项目。


我本应该在滚动时获取接下来的10条记录,以此类推...但是只获取到了前10条记录,请检查我上面更新的代码。 - Oreo

0
问题1: 在为RecyclerView设置LayoutManager之前,您已经创建了一个新的mAdapter实例。因此,在DataAdapter中添加ScrollListener的构造函数代码未被执行,因为recyclerView.getLayoutManager()返回null
    if (recyclerView.getLayoutManager() instanceof LinearLayoutManager){
        // code to add ScrollListener is never executed
    }
修复:先为RecyclerView设置LayoutManager,然后按以下方式创建适配器:
    // use a linear layout manager
    mRecyclerView.setLayoutManager(mLayoutManager);
    mAdapter = new DataAdapter(temArray, mRecyclerView);
    // set the adapter object to the Recyclerview
    mRecyclerView.setAdapter(mAdapter);

问题-2:您使用temArray创建了DataAdapter,但在onLoadMore()中,您正在使用studentList添加/删除新项目,因为studentList未与mAdapter绑定,所以您的更改不会反映在UI中。

解决方案:temArray声明为类级变量,并使用temArray来操作项目。

 //class variable
 private ArrayList<Student> temArray = new ArrayList<Student>();

 handler.postDelayed(new Runnable() {
            @Override public void run() {
              //   remove progress item
              temArray.remove(temArray.size() - 1);
              mAdapter.notifyItemRemoved(temArray.size());
              //add items one by one
              int start = temArray.size();
              int end = start + 10;
              if(end<=studentList.size()){
                temArray.addAll(studentList.subList(start,end));
              }   
              mAdapter.setLoaded();
            }
          }, 2000);

-1
        mAdapter.setOnLoadMoreListener(new OnLoadMoreListener() {
                    @Override
                    public void onLoadMore() {
                        //add null , so the adapter will check view_type and show progress bar at bottom
                        studentList.add(null);
                        mAdapter.notifyItemInserted(studentList.size() - 1);

                        handler.postDelayed(new Runnable() {
                            @Override
                            public void run() {
                                //   remove progress item
                                studentList.remove(studentList.size() - 1);
                                mAdapter.notifyItemRemoved(studentList.size());
                                //add items one by one
                                int start = studentList.size();
                                int end = start + 10;

                                for (int i = start + 1; i < end; i++) {
   // studentList.add();
                                    mAdapter.notifyItemInserted(studentList.size());
                                }
                                mAdapter.setLoaded();
                               //or you can add all at once but do not forget to call mAdapter.notifyDataSetChanged();
                            }
                        }, 2000);
                    }
                });

好的,所以你插入了进度条,然后又将其删除了,但是你从未插入下一个要显示的学生... 类似于studentList.add();

希望这解决了你的问题...祝你好运。


-1
for (int i = start + 1; i < end; i++) {
    studentList.add(add data here) ;
    mAdapter.notifyItemInserted(studentList.size());
}

我认为在调用notifyItemInserted之前,您必须将数据添加到列表中。 - Ankur1994a

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