- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
getAdapterPosition()
和 getLayoutPosition
有什么区别?
https://proandroiddev.com/difference-between-position-getadapterposition-and-getlayoutposition-in-recyclerview-80279a2711d1 .我已经阅读了这篇文章,但并不真正理解有关 getLayoutPosition()
请用更简单的术语解释它及其背后的过程。
在构建应用时,您什么时候会使用其中之一?
最佳答案
getAdapterPosition()
在任何 notify*()
调用后立即返回当前 ViewHolder
对象的更新位置。
getLayoutPosition()
在调度新布局后稍后返回更新值。
getAdapterPosition()
返回的适配器位置将在您在 Adapter
变化:
notifyItemInserted(...)
notifyItemRemoved(...)
...
notifyDataSetChanged()
例如,如果我们查看 notifyItemInserted()
,那么它将调用:
notifyItemInserted(int position)
|
v
...
|
v
public void onItemRangeInserted(int positionStart, int itemCount) {
assertNotInLayoutOrScroll(null);
if (mAdapterHelper.onItemRangeInserted(positionStart, itemCount)) {
triggerUpdateProcessor();
}
}
mAdapterHelper.onItemRangeInserted(positionStart, itemCount)
基本上会更新用于检索当前适配器位置的 mPendingUpdates
。下一次调用 triggerUpdateProcessor()
将请求布局更新,但更多信息将在下一节中介绍。
现在,如果您随后立即为 ViewHolder
调用 getAdapterPosition()
,它将返回更新后的位置。但是,getLayoutPosition()
仍将返回旧值。
triggerUpdateProcessor
将调用 requestLayout()
并返回。这将安排 RecyclerView
的布局更新(稍后将调用 onLayout
以指示布局更改):
public void onItemRangeInserted(int positionStart, int itemCount) {
assertNotInLayoutOrScroll(null);
if (mAdapterHelper.onItemRangeInserted(positionStart, itemCount)) {
triggerUpdateProcessor();
}
}
虽然这需要一些时间,但是在触发 onLayout
回调并调用 dispatchLayout()
之后更新将成为可能:
protected void onLayout(boolean changed, int l, int t, int r, int b) {
...
dispatchLayout();
...
}
基本上,在onLayout
返回后,布局位置已经更新。
这可以用一个简单的 RecyclerView
来说明,其中在列表顶部添加新元素时调用 notifyInserted()
:
...
public void addElementToHead() {
elements.add(0, "New Element");
notifyItemInserted(0);
Toast.makeText(context, "Adapter position : " + currentFirstHolder.getAdapterPosition() +
", Layout position : " + currentFirstHolder.getLayoutPosition(), Toast.LENGTH_LONG).show();
}
...
在这种情况下,currentFirstHolder
是一个 ViewHolder
,对应于适配器中的初始第一个元素。但是,在将新元素添加为第一个元素后,currentFirstHolder
的位置必须更改为 1
。所以,getAdapterPosition()
是 1
,而 getLayoutPosition()
仍然是 0
因为 RecyclerView
尚未完成布局更新。
在这里,我记录了 ViewHolder
的内容,以显示在最终调用 onLayout
之后它是如何变化的:
...
@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
super.onLayout(changed, l, t, r, b);
ViewHolder viewHolder = findViewHolderForAdapterPosition(0);
Toast.makeText(getContext(), "From RecyclerView#onLayout - Adapter position : " +
viewHolder.getAdapterPosition() + ", Layout position : " + viewHolder.getLayoutPosition(), Toast.LENGTH_LONG).show();
}
...
因此,它将如下所示(它将在第一个 toast 消失后立即显示):
现在,完整的相关代码可以在下面看到。为简洁起见,我没有包含布局和其他文件。
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
RecyclerViewAdapter adapter = new RecyclerViewAdapter(this);
MyRecyclerView recyclerView = findViewById(R.id.recycler_view);
recyclerView.setAdapter(adapter);
recyclerView.setLayoutManager(new LinearLayoutManager(this));
Button button = findViewById(R.id.add_element_button);
button.setOnClickListener(v -> adapter.addElementToHead());
}
private static class RecyclerViewAdapter extends RecyclerView.Adapter<RecyclerViewAdapter.RecyclerViewHolder> {
private final Context context;
private final List<String> elements = new ArrayList<>();
private RecyclerViewAdapter(Context context) {
this.context = context;
for (int i = 0; i < 1; i++) {
elements.add("Element " + i);
}
}
public void addElementToHead() {
elements.add(0, "New Element");
notifyItemInserted(0);
Toast.makeText(context, "Adapter position : " + currentFirstHolder.getAdapterPosition() +
", Layout position : " + currentFirstHolder.getLayoutPosition(), Toast.LENGTH_LONG).show();
}
private RecyclerViewHolder currentFirstHolder;
@NonNull
@Override
public RecyclerViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
View item = LayoutInflater.from(context).
inflate(R.layout.list_item, parent, false);
return new RecyclerViewHolder(item);
}
@Override
public void onBindViewHolder(@NonNull RecyclerViewHolder holder, int position) {
holder.textView.setText(elements.get(position));
if (position == 0) {
currentFirstHolder = holder;
}
}
@Override
public int getItemCount() {
return elements.size();
}
public static class RecyclerViewHolder extends RecyclerView.ViewHolder {
private final TextView textView;
public RecyclerViewHolder(@NonNull View itemView) {
super(itemView);
this.textView = itemView.findViewById(R.id.element_view);
}
}
}
}
public class MyRecyclerView extends RecyclerView {
public MyRecyclerView(@NonNull Context context) {
super(context);
}
public MyRecyclerView(@NonNull Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
}
public MyRecyclerView(@NonNull Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
super.onLayout(changed, l, t, r, b);
ViewHolder viewHolder = findViewHolderForAdapterPosition(0);
Toast.makeText(getContext(), "From RecyclerView#onLayout - Adapter position : " +
viewHolder.getAdapterPosition() + ", Layout position : " + viewHolder.getLayoutPosition(), Toast.LENGTH_LONG).show();
}
}
关于java - getLayoutPosition() 和 getAdapterPosition() 有什么区别?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/66879093/
getAdapterPosition() 和 getLayoutPosition 有什么区别? https://proandroiddev.com/difference-between-positio
由于新的支持库版本 (22.x),RecyclerView.ViewHolder 类的 getPosition() 方法已被弃用,以代替话题。我并没有真正从阅读文档中得到区别。有人可以用外行的话解释一
我真的不知道为什么我不能在我的类中使用 getLayoutPosition(); 或 getAdapterPosition(); 方法。 class MyViewHolder extends
我是一名优秀的程序员,十分优秀!