今天在写一个RecyclerView的Demo,大致的状况就是请求网络分页加载数据,解析成bean然后填到列表里,展示瀑布流出来.但是写完之后列表却一直都是空的,但是断点里也能清楚地看到数据解析成功,被设置到了adapter中,反复看了好久,最后还是在同事的指点下,跳出了坑,其实问题是Java引用的问题,我先把错误的代码贴出来.大家观摩一下.
Adapter:
public class RecyclerViewAdapter extends RecyclerView.Adapter<PostViewHolder> {
private Context context;
public List<PostsEntity> list = new ArrayList<>();
public RecyclerViewAdapter(Context context, List<PostsEntity> list) {
this.context = context;
this.list = list;
}
@Override
public PostViewHolder onCreateViewHolder(ViewGroup viewGroup, int viewType) {
PostViewHolder holder = new PostViewHolder(LayoutInflater.from(
context).inflate(R.layout.post_detail_item, viewGroup, false));
return holder;
}
@Override
public void onBindViewHolder(PostViewHolder viewHolder, int position) {
//填充界面等操作...
}
@Override
public int getItemCount() {
return list == null ? 0 : list.size();
}
}
Activity:
public class HomeActivity extends AppCompatActivity {
@Bind(R.id.recyclerview)
RecyclerView recyclerview;
private List<PostsEntity> originalList;
private RecyclerViewAdapter adapter;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_home);
ButterKnife.bind(this);
init();
loadData();
}
private void init() {
list = new ArrayList<>();
adapter = new RecyclerViewAdapter(HomeActivity.this, originalList);
LinearLayoutManager mManager = new LinearLayoutManager(HomeActivity.this, LinearLayoutManager.VERTICAL, false);
recyclerview.setAdapter(adapter);
recyclerview.setLayoutManager(mManager);
}
private void loadData() {
OkHttpUtils.get().url(URL).build().execute(new StringCallback() {
@Override
public void onError(Call call, Exception e) {
}
@Override
public void onResponse(final String response) {
Gson gson = new Gson();
final PostBean bean = gson.fromJson(response, PostBean.class);
assembleData(bean);
}
});
}
private void assembleData(PostBean bean) {
originalList = bean.getPosts();<span style="white-space:pre"> </span>//问题就出现在了这里!!!
adapter.notifyDataSetChanged();
}
}
好了直接看出问题的可以打我了
,没看出来的往下看,听我缩.
讲解原因之前,先介绍一下Java的内存模型,其实在自己刚开始学习Java的时候(一年多之前)记过这样一条笔记:
但是,如果当初能够真正的理解这些概念,也不至于有今天的事情,还是那句话:纸上得来终觉浅,绝知此事要躬行.
上面的话大致可以这样理解,Book book = new Book();这句话其实创建了两个变量:book为引用变量,存储在堆栈中;new出来的Book()为值变量,存在堆中.book的值就是Book()的地址.这是分析上面问题的先决条件.
下面再上一张我自己画的关于这段出错程序的图,
上图一共分为6部,三个区域(从左至右为Activity的栈空间, Java的堆空间,Adapter的栈空间),我们一步一步看.
1.在Activity的onCreate()方法中我们初始化一个空的originalList,用来存放接下来从服务端获取的bean.
2.我们在new RecyclerViewAdapter的时候,把originalList以构造方法参数的形式传进去.
3.在Adapter的构造方法内部,this.list = list;这一行就是第三步.
4.在loadData()方法中我们请求到了真正的数据bean.getPosts(),这是一个以PostEntity为元素的列表.
5和6.在我们出错的那一行,originalList = bean.getPosts();此时original作为一个引用变量,它的值变了,不再指向上面的那块内存区域,从而指向下面的那块内存区域.
可是这时候我们的adapter内的list还是指向原来的内存区域,那块区域里还是空的,自然RecyclerView就是空的了,不展示任何东西.所以问题症结所在也就一目了然了.
下面贴一下正确的做法:
private void assembleData(PostBean bean) {
list.clear();
list.addAll(bean.getPosts())
adapter.notifyDataSetChanged();
}
最后要做一下总结,出现这个问题还是因为自己的基础不过关,当初是记过笔记,但大多还是流于表面形式,没有在编码实践中有所体会.正所谓"基础不牢,地动山摇",现在踩得每一个坑都是自己以前埋下的.
最后感谢一下今天帮我出坑的我司小伙伴(技术没的说,虐我无压力):@写代码的猴子.(知乎网红,我会卵缩么?)
博客:写代码的猴子
github:laobie
知乎:Jaeger
相关问题链接请戳:http://stackoverflow.com/questions/21598621/notifydatasetchanged-not-working