- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
更新:其中一个问题解决了:现在updateList
解决了,问题是我把mAdapter
定义为RecyclerView .Adapter
而不是 MyAdapter
。但是现在即使我正在获取数据,列表中也没有显示任何内容,它是空的
--------------------原帖---------------- --
我想使用 DiffUtil
更新我的 RecyclerView
以防止重复。
我有 4 个类:User 类、我设置数据的 Activity
类、Adapter
类和 DiffUtil
类。我不确定我是否正确组合了所有这 4 个。
这是用户类:
public class User {
private String mUserId;
private Uri mImageUrl;
public User(String userId, String imageUrl) {
mUserId = userId;
mImageUrl = Uri.parse(imageUrl);
}
public String getUserId() {
return mUserId;
}
public Uri getImageUrl() {
return mImageUrl;
}
}
这就是我动态设置数据的方式(我不断从服务器获取新的 Json
数组,其中包含要显示的用户 ID,然后我从 Firebase
存储中设置用户图像): (这是一个由 onClick 监听器调用的函数:)
这是 fragment 中的方法调用:
button.setOnClickListener(new View.OnClickListener() {
public void onClick(View v) {
updateUsersList();
}
});
这是函数:
private void updateUsersList() {
@Override
public void onResponse(JSONArray response) { // the JSON ARRAY response of user ids ["uid1", "uid334", "uid1123"]
myDataset.clear(); // clear dataset to prevent duplicates
for (int i = 0; i < response.length(); i++) {
try {
String userKey = response.get(i).toString(); // the currently iterated user id
final DatabaseReference rootRef = FirebaseDatabase.getInstance().getReference();
DatabaseReference userKeyRef = rootRef.child("users").child(userKey); // reference to currently iterated user
ValueEventListener listener = new ValueEventListener() {
@Override
public void onDataChange(DataSnapshot dataSnapshot) {
myDataset.add(new User(dataSnapshot.getKey(), dataSnapshot.child("imageUrl").getValue().toString())); //add new user: id and image url
mAdapter.updateList(myDataset); // cannot resolve this method, why?
}
@Override
public void onCancelled(@NonNull DatabaseError databaseError) {
Log.d(TAG, databaseError.getMessage());
}
};
userKeyRef.addListenerForSingleValueEvent(listener);
}
catch (JSONException e) { Log.d(TAG, "message " + e); }
}
}
这是我的 DiffUtil
类的样子:
public class MyDiffUtilCallBack extends DiffUtil.Callback{
ArrayList<User> oldUsers;
ArrayList<User> newUsers;
public MyDiffUtilCallBack(ArrayList<User> newUsers, ArrayList<User> oldUsers) {
this.newUsers = newUsers;
this.oldUsers = oldUsers;
}
@Override
public int getOldListSize() {
return oldUsers.size();
}
@Override
public int getNewListSize() {
return newUsers.size();
}
@Override
public boolean areItemsTheSame(int oldItemPosition, int newItemPosition) {
return oldUsers.get(oldItemPosition).getUserId().equals( newUsers.get(newItemPosition).getUserId());
}
@Override
public boolean areContentsTheSame(int oldItemPosition, int newItemPosition) {
return oldUsers.get(oldItemPosition).equals(newUsers.get(newItemPosition));
}
@Nullable
@Override
public Object getChangePayload(int oldItemPosition, int newItemPosition) {
//you can return particular field for changed item.
return super.getChangePayload(oldItemPosition, newItemPosition);
}
}
这是我的适配器:
public class MyAdapter extends RecyclerView.Adapter<MyAdapter.MyViewHolder> {
private ArrayList<User> mDataset;
private MyViewHolder myHolder;
private User user;
public static class MyViewHolder extends RecyclerView.ViewHolder {
public TextView singleItemTextView;
public ImageView singleItemImage;
public View layout;
public ConstraintLayout constraintLayout;
public MyViewHolder(View v) {
super(v);
layout = v;
singleItemImage = (ImageView) v.findViewById(R.id.icon);
singleItemTextView = (TextView) v.findViewById(R.id.singleitemtv);
constraintLayout = (ConstraintLayout) v.findViewById(R.id.nbConstraintLayout);
}
}
// Provide a suitable constructor (depends on the kind of dataset)
public MyAdapter(ArrayList<User> myDataset) {
mDataset = myDataset;
}
// Create new views (invoked by the layout manager)
@Override
public MyAdapter.MyViewHolder onCreateViewHolder(ViewGroup parent,
int viewType) {
View v = LayoutInflater.from(parent.getContext())
.inflate(R.layout.nb_image_view, parent, false);
MyViewHolder vh = new MyViewHolder(v);
return vh;
}
@Override
public void onBindViewHolder(final MyViewHolder holder, final int position) {
myHolder = holder;
user = mDataset.get(position);
Uri userImage = user.getImageUrl();
myHolder.singleItemTextView.setText(user.getUserId());
Glide.with(myHolder.itemView.getContext() /* context */)
.load(userImage)
.into(myHolder.singleItemImage);
myHolder.constraintLayout.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Context context = v.getContext();
Intent intent = new Intent(v.getContext(), DisplayUserActivity.class);
context.startActivity(intent);
}
});
}
public void updateList(ArrayList<User> newList) {
DiffUtil.DiffResult diffResult = DiffUtil.calculateDiff(new MyDiffUtilCallBack(this.mDataset, newList));
diffResult.dispatchUpdatesTo(this);
}
}
我不确定我是否正确组合了所有类(我第一次使用 DiffUtil
),而且我还得到了 cannot resolve method updateList(?)
我做错了什么?
这就是我在 Fragment 中定义 mAdapter 的方式:
public class MyFragment extends Fragment {
private ArrayList<User> myDataset;
private RecyclerView.Adapter mAdapter;
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
// Inflate the layout for this fragment
rootView = inflater.inflate(R.layout.fragment_lks, container, false);
mRecyclerView = (RecyclerView) rootView.findViewById(R.id.my_recycler_view);
myDataset = new ArrayList<User>();
mAdapter = new MyAdapter(myDataset);
最佳答案
问题出在 mAdapter
的定义上。您将其定义为 RecyclerView.Adapter
,它是您的 MyAdapter
的父类(super class),并且不包含 updateList()
。您应该按如下方式更改它:
private MyAdapter mAdapter;
2019 年 1 月 13 日更新:
我已经用 AsyncListDiffer
修改了你的适配器,它异步计算差异然后将它应用到适配器。
MyAdapter.java
import android.content.Context;
import android.content.Intent;
import android.net.Uri;
import android.support.annotation.NonNull;
import android.support.constraint.ConstraintLayout;
import android.support.v7.recyclerview.extensions.AsyncListDiffer;
import android.support.v7.util.DiffUtil;
import android.support.v7.widget.RecyclerView;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.TextView;
import com.bumptech.glide.Glide;
import java.util.List;
public class MyAdapter extends RecyclerView.Adapter<MyAdapter.MyViewHolder> {
private AsyncListDiffer<User> mAsyncListDiffer;
public static class MyViewHolder extends RecyclerView.ViewHolder {
public TextView singleItemTextView;
public ImageView singleItemImage;
public View layout;
public ConstraintLayout constraintLayout;
public MyViewHolder(View v) {
super(v);
layout = v;
singleItemImage = (ImageView) v.findViewById(R.id.icon);
singleItemTextView = (TextView) v.findViewById(R.id.singleitemtv);
constraintLayout = (ConstraintLayout) v.findViewById(R.id.nbConstraintLayout);
}
}
// Provide a suitable constructor (depends on the kind of dataset)
public MyAdapter() {
DiffUtil.ItemCallback<User> diffUtilCallback = new DiffUtil.ItemCallback<User>() {
@Override
public boolean areItemsTheSame(@NonNull User newUser, @NonNull User oldUser) {
return newUser.getUserId().equals(oldUser.getUserId());
}
@Override
public boolean areContentsTheSame(@NonNull User newUser, @NonNull User oldUser) {
return newUser.equals(oldUser);
}
};
mAsyncListDiffer = new AsyncListDiffer<>(this, diffUtilCallback);
}
// Create new views (invoked by the layout manager)
@Override
public MyAdapter.MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View v = LayoutInflater.from(parent.getContext()).inflate(R.layout.nb_image_view, parent, false);
MyViewHolder vh = new MyViewHolder(v);
return vh;
}
@Override
public void onBindViewHolder(final MyViewHolder holder, final int position) {
User user = mAsyncListDiffer.getCurrentList().get(position);
Uri userImage = user.getImageUrl();
holder.singleItemTextView.setText(user.getUserId());
Glide.with(holder.itemView.getContext() /* context */)
.load(userImage)
.into(holder.singleItemImage);
holder.constraintLayout.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Context context = v.getContext();
Intent intent = new Intent(v.getContext(), DisplayUserActivity.class);
context.startActivity(intent);
}
});
}
@Override
public int getItemCount() {
return mAsyncListDiffer.getCurrentList().size();
}
public void updateList(List<User> newList) {
mAsyncListDiffer.submitList(newList);
}
}
用户.java
public class User {
private String mUserId;
private Uri mImageUrl;
public User(String userId, String imageUrl) {
mUserId = userId;
mImageUrl = Uri.parse(imageUrl);
}
public String getUserId() {
return mUserId;
}
public Uri getImageUrl() {
return mImageUrl;
}
@Override
public boolean equals(Object other) {
if (other instanceof User) {
User user = (User) other;
return mUserId.equals(user.getUserId()) && mImageUrl.equals(user.getImageUrl());
} else {
return false;
}
}
}
关于java - 我的 DiffUtil 实现有什么问题?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54162101/
我们正在尝试对包含来自 sqlite 数据库的数据的 ArrayList 进行排序。 我可以通过添加到查询“ORDER BY”+ Col_IDBG+“DESC 来对列表进行排序 除了 Recycler
我通常在需要时将 diffutil 与 Recyclerview 一起使用。现在我有这样的情况,我从后端得到的项目是这样的: data class CarouselItem(var url: Stri
我正在尝试使用DiffUtil来更新RecyclerView。我有一个类 DataProvider ,它是一个单例类,它保存包含对象的列表。 每次我需要更新RecyclerView时,我都会修改Dat
我有一个 RecyclerView,其中的元素通过 WebSocket 更新。此更新可能会影响列表中不同位置的多个元素。我想用 DiffUtil更新 RecycleView 的元素。套接字更新本身不包
DiffUtils.dispatchUpdatesTo(adapter) 没有更新我在 recyclerView 中的数组大小 我的 result 的大小大于当前的 itemList,但只显示当前的大
我为 recyclerView 制作了一个适配器,并使用 DiffUtil 以更漂亮的方式显示列表的更新。我使用 google codelabs 作为引用。但该列表并未从以下代码更新。请帮忙 clas
我正在使用 diff util 来提高我的 recyclerview 的性能,而不是调用 notifyDataSetChanged()。 recyclerview 有一个带有一些芯片的标题,可以通过
我有一个具有 DiffUtil 功能的 RecyclerView 适配器。 我想在 DiffUtil 完成其魔法时使用 DataObserver 通知 fragment 。但是,看起来 DiffUti
我的回收站 View 中有无限滚动,因此,它会在有新数据时更新。我正在使用 DiffUtil 更新 recyclerview 中的数据。 DiffUtil 确实会更新数据,但是只要有更新数据,recy
一段时间以来,我一直被困在一个相当简单的问题上。出于某种原因,我无法在 StackOverflow 上找到这方面的答案,所以我想我应该直接寻求帮助。 问题很简单,我有一个按日期排序的交易对象列表。我正
我正在使用下面的代码来测试 DiffUtil实现。根据文档- DiffUtil is a utility class that can calculate the difference between
更新:其中一个问题解决了:现在updateList解决了,问题是我把mAdapter定义为RecyclerView .Adapter 而不是 MyAdapter。但是现在即使我正在获取数据,列表中也没
目前我尝试在我的项目中使用DiffUtil来通知RecyclerView的项目。 与此同时,我编写了一个可以添加页眉和页脚的自定义适配器。我使用了这个适配器并添加了一个加载更多的页脚,这样我的列表在向
我对 Android 还很陌生,目前正在开发我的第一个基于 RecyclerView 的严肃应用程序。当我插入或更新现有行时,我注意到我的应用程序中有一些有问题的行为:即使我在添加或 notifyIt
我使用 ListAdapter 的绑定(bind)和 DiffUtil.ItemCallback 的定义。删除项目(至少 2 个)时出现 IndexOutOfBoundsException。列表的更新
每次我调用无效数据时,我的 DIFF_UTIL 都不会被使用。日志未显示,整个列表已更新为新数据,导致屏幕移动位置等。不确定这里的问题是什么。 我有PagedListAdapter与 LiveData
我无法将 Kotlin Flows 和异步 DiffUtil 放在一起。 我的 RecyclerView.Adapter 中有这个函数,它在计算线程上计算 DiffUtil 并将更新发送到主线程上的
我是 Android 的新手,我正在尝试使用新架构组件实现条码阅读器场景。 每次读取条码时,如果条码不在列表中,我想更新 ViewModel 中的列表,添加新元素,否则增加数量。 以下解决方案有效,但
我遇到了一个有趣的情况,我的适配器有 2 种 View 类型 - HEADER 和 ITEM。 header View 始终位于位置 0。更新数据时,我使用 DiffUtil 获取数据的差异。为了分派
我的 diffcallback areContentsTheSame(oldItem: ItemModel, newItem: ItemModel) 总是接收相同的内容。我用状态来检查,但每次状态都是
我是一名优秀的程序员,十分优秀!