Android 探究RecyclerView onViewAttachedToWindow 触发时机

转载请标明出处:http://blog.csdn.net/zhaoyanjun6/article/details/125759325
本文出自【赵彦军的博客】

RecyclerView 基本使用

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<androidx.appcompat.widget.LinearLayoutCompat xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <androidx.recyclerview.widget.RecyclerView
        android:id="@+id/fruit_recycler_view"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />

</androidx.appcompat.widget.LinearLayoutCompat>

item_layout.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="200dp"
    android:background="#ddd"
    android:layout_marginTop="10dp"
    android:orientation="vertical">

    <TextView
        android:id="@+id/fruit_name"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:textSize="30sp"
        android:layout_gravity="center"
        android:gravity="center"
        android:textColor="#aaffcc" />
</LinearLayout>

Fruit

public class Fruit {
    private String fruitName;

    public String getFruitName() {
        return fruitName;
    }

    public Fruit(String fruitName) {
        this.fruitName = fruitName;
    }

    //获取数据
    public static List<Fruit> initFruitList() {
        List<Fruit> fruitList = new ArrayList<>();
        for (int i = 0; i < 20; i++) {
            Fruit apple = new Fruit("Item " + i);
            fruitList.add(apple);
        }
        return fruitList;
    }
}

FruitAdapter

public class FruitAdapter extends RecyclerView.Adapter<FruitAdapter.ViewHolder> {

    private List<Fruit> mFruitList;

    public FruitAdapter(List<Fruit> fruitList) {
        mFruitList = fruitList;
    }

    @Override
    public ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
        View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_layout, parent, false);
        ViewHolder viewHolder = new ViewHolder(view);
        return viewHolder;
    }

    @Override
    public void onBindViewHolder(@NonNull ViewHolder viewHolder, int position) {
        Fruit fruit = mFruitList.get(position);
        viewHolder.fruitName.setText(fruit.getFruitName());
    }

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

    public class ViewHolder extends RecyclerView.ViewHolder {
        TextView fruitName;

        public ViewHolder(@NonNull View itemView) {
            super(itemView);
            fruitName = itemView.findViewById(R.id.fruit_name);
        }
    }

    @Override
    public void onDetachedFromRecyclerView(@NonNull RecyclerView recyclerView) {
        super.onDetachedFromRecyclerView(recyclerView);
    }

    @Override
    public void onAttachedToRecyclerView(@NonNull RecyclerView recyclerView) {
        super.onAttachedToRecyclerView(recyclerView);
    }

    @Override
    public void onViewAttachedToWindow(@NonNull ViewHolder holder) {
        super.onViewAttachedToWindow(holder);
    }

    @Override
    public void onViewDetachedFromWindow(@NonNull ViewHolder holder) {
        super.onViewDetachedFromWindow(holder);
    }
}

MainActivity

class MainActivity : AppCompatActivity() {

    private lateinit var binding: ActivityMainBinding
    private var adapter: FruitAdapter? = null
    private var list = mutableListOf<Fruit>()

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = ActivityMainBinding.inflate(layoutInflater)
        setContentView(binding.root)

        list = Fruit.initFruitList()
        adapter = FruitAdapter(list)

        binding.fruitRecyclerView.layoutManager = LinearLayoutManager(this)
        binding.fruitRecyclerView.adapter = adapter

    }
}

FruitAdapter

在上面,FruitAdapter 有四个方法

   @Override
    public void onDetachedFromRecyclerView(@NonNull RecyclerView recyclerView) {
        super.onDetachedFromRecyclerView(recyclerView);
    }

    @Override
    public void onAttachedToRecyclerView(@NonNull RecyclerView recyclerView) {
        super.onAttachedToRecyclerView(recyclerView);
    }

    @Override
    public void onViewAttachedToWindow(@NonNull ViewHolder holder) {
        super.onViewAttachedToWindow(holder);
    }

    @Override
    public void onViewDetachedFromWindow(@NonNull ViewHolder holder) {
        super.onViewDetachedFromWindow(holder);
    }

Adapter中的 onAttachedToRecyclerView

onAttachedToRecyclerViewonDetachedFromRecyclerView 这两个方法是对于 adapter 来说的,当 adapter recyclerView 进行绑定的时候会回调 onAttachedToRecyclerView

什么时候会回调 onDetachedFromRecyclerView

答案:recyclerView 已经有 adapter 了,又绑定了一个新的 adapter ,就会执行 onDetachedFromRecyclerView

对于一个 adapter 实例,onAttachedToRecyclerViewonDetachedFromRecyclerView 这两个方法只会回调一次。

源码分析:

onAttachedToRecyclerViewonDetachedFromRecyclerViewRecyclerView 都是空方法。

在这里插入图片描述

首先看看 RecyclerViewsetAdapter 方法

在这里插入图片描述

setAdapter 方法里,调用 setAdapterInternal 方法

在这里插入图片描述

设置一个新的 adapter , 如果 recyclerView 已经绑定过 adapter

那么就对于老的adapter调用 onDetachedFromRecyclerView(this)

新的 adapter 调用 onAttachedToRecyclerView(this)

Adapter中的 onViewAttachedToWindow

   @Override
    public void onViewAttachedToWindow(@NonNull ViewHolder holder) {
        super.onViewAttachedToWindow(holder);
        int position = holder.getLayoutPosition();
        Log.d("yu--", "attach " + position);
    }

    @Override
    public void onViewDetachedFromWindow(@NonNull ViewHolder holder) {
        super.onViewDetachedFromWindow(holder);
        int position = holder.getLayoutPosition();
        Log.d("yu--", "detach " + position);
    }

RecyclerView 本质上也是一个 ViewGroup,那么它的 Item 要显示出来,自然需要 addView() 进来,移出的时候,当然也要 removeView() 出去,所以对应的自然是 onViewAttachedToWindow()onViewAttachedToWindow() 了。

所以在特定场景下,可以通过这两个回调来解决少量 Item 移出屏幕,移进屏幕所需要的工作。为什么说特定场景下呢,由于假如调用了 notifyDataSetChanged() 方法的话,会触发当前在屏幕中的所有 Item 的 onViewAttachedToWindow()

当第一次 setAdapter 的时候,屏幕内所有 item 会调用 onViewAttachedToWindow(ViewHolder holder) , 日志如下:

yu--: attach 0
yu--: attach 1
yu--: attach 2
yu--: attach 3
yu--: attach 4
yu--: attach 5
yu--: attach 6

调用 notifyDataSetChanged ,屏幕类所有的 item 会先执行 onViewDetachedFromWindow , 然后执行 onViewAttachedToWindow

yu--: detach 6
yu--: detach 5
yu--: detach 4
yu--: detach 3
yu--: detach 2
yu--: detach 1
yu--: detach 0
yu--: attach 0
yu--: attach 1
yu--: attach 2
yu--: attach 3
yu--: attach 4
yu--: attach 5
yu--: attach 6

综上,Adapter 的 onViewAttachedToWindow 适合做 item 曝光埋点,但是要注意,这个方法可能会执行多次。

瀑布流 StaggeredGridLayoutManager 设置第一个 item 占据一整行

   @Override
    public void onViewAttachedToWindow(@NonNull ViewHolder holder) {
        super.onViewAttachedToWindow(holder);
        int position = holder.getLayoutPosition();
        StaggeredGridLayoutManager.LayoutParams layoutParams = (StaggeredGridLayoutManager.LayoutParams) holder.itemView.getLayoutParams();
        if (position == 0) {
            //占据一整行
            layoutParams.setFullSpan(true);
        } else {
            layoutParams.setFullSpan(false);
        }
    }

效果图:

在这里插入图片描述

  • 2
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android开发中,RecyclerView是一种常用的列表展示件,它可以用于展示大量数据,并且支持自定义布局和交互效果。实现RecyclerView的加载更多功能可以通过以下步骤来完成: 1. 首先,在你的布局文件中添加RecyclerView控件,并为其指定一个唯一的id,例如: ```xml <androidx.recyclerview.widget.RecyclerView android:id="@+id/recyclerView" android:layout_width="match_parent" android:layout_height="match_parent" /> ``` 2. 在你的Activity或Fragment中,找到RecyclerView控件并创建一个适配器(Adapter)来管理数据的展示。适配器需要继承RecyclerView.Adapter类,并实现其中的几个方法,例如: ```java public class MyAdapter extends RecyclerView.Adapter<MyAdapter.ViewHolder> { // 数据集合 private List<Data> dataList; // 构造方法 public MyAdapter(List<Data> dataList) { this.dataList = dataList; } // 创建ViewHolder @Override public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_layout, parent, false); return new ViewHolder(view); } // 绑定ViewHolder @Override public void onBindViewHolder(ViewHolder holder, int position) { Data data = dataList.get(position); // 设置数据到ViewHolder中的控件上 holder.textView.setText(data.getText()); } // 获取数据数量 @Override public int getItemCount() { return dataList.size(); } // ViewHolder类 public static class ViewHolder extends RecyclerView.ViewHolder { TextView textView; public ViewHolder(View itemView) { super(itemView); textView = itemView.findViewById(R.id.textView); } } } ``` 3. 在你的Activity或Fragment中,设置RecyclerView的布局管理器(Layout Manager)和适配器(Adapter),例如: ```java RecyclerView recyclerView = findViewById(R.id.recyclerView); recyclerView.setLayoutManager(new LinearLayoutManager(this)); recyclerView.setAdapter(new MyAdapter(dataList)); ``` 4. 实现加载更多的功能,可以通过监听RecyclerView的滑动事件来实现。当用户滑动到列表底部时,触发加载更多的操作。具体实现可以参考以下步骤: - 在适配器中添加一个方法用于加载更多数据,例如: ```java public void loadMoreData(List<Data> moreDataList) { dataList.addAll(moreDataList); notifyDataSetChanged(); } ``` - 在Activity或Fragment中,为RecyclerView添加滑动监听器,并在滑动到底部时触发加载更多的操作,例如: ```java recyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() { @Override public void onScrolled(@NonNull RecyclerView recyclerView, int dx, int dy) { super.onScrolled(recyclerView, dx, dy); LinearLayoutManager layoutManager = (LinearLayoutManager) recyclerView.getLayoutManager(); int visibleItemCount = layoutManager.getChildCount(); int totalItemCount = layoutManager.getItemCount(); int firstVisibleItemPosition = layoutManager.findFirstVisibleItemPosition(); if ((visibleItemCount + firstVisibleItemPosition) >= totalItemCount && firstVisibleItemPosition >= 0) { // 滑动到底部,执行加载更多的操作 // 调用适配器的加载更多方法 adapter.loadMoreData(moreDataList); } } }); ``` 这样,当用户滑动到RecyclerView的底部时,就会触发加载更多的操作,新的数据会被添加到适配器中,并通过调用`notifyDataSetChanged()`方法来更新列表的显示。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值