提高Android RecyclerView性能的技巧

一、使用ViewHolder提升性能

在Android 5.0以前,RecyclerView使用ViewHolder来重用布局并减少内存消耗。ViewHolder是一个包含视图组件的容器,可以提高视图重用的效率。我们可以通过继承RecyclerView.ViewHolder来创建ViewHolder。

class MyViewHolder extends RecyclerView.ViewHolder {
    TextView name;
    TextView age;
    public MyViewHolder(View view) {
        super(view);
        name = (TextView) view.findViewById(R.id.name);
        age = (TextView) view.findViewById(R.id.age);
    }
}

ViewHolder的使用可以减少inflate布局的次数,从而提高RecyclerView的性能。在onBindViewHolder()方法中,我们可以将数据绑定到ViewHolder中的视图组件上。

@Override
public void onBindViewHolder(MyViewHolder holder, int position) {
    holder.name.setText(mData.get(position).getName());
    holder.age.setText(mData.get(position).getAge());
}

二、使用DiffUtil优化数据更新

RecyclerView经常需要更新数据,而DiffUtil可以提高数据更新的效率。DiffUtil是一个工具类,用于计算两个数据集之间的差异。在数据更新时,我们可以使用DiffUtil来计算数据集差异,从而只更新变化的部分,减少不必要的重绘和动画效果。

class MyDiffCallback extends DiffUtil.Callback {
    private List oldList;
    private List newList;
    public MyDiffCallback(List oldList, List newList) {
        this.oldList = oldList;
        this.newList = newList;
    }
    @Override
    public int getOldListSize() {
        return oldList.size();
    }
    @Override
    public int getNewListSize() {
        return newList.size();
    }
    @Override
    public boolean areItemsTheSame(int oldItemPosition, int newItemPosition) {
        return oldList.get(oldItemPosition).getId() == newList.get(newItemPosition).getId();
    }
    @Override
    public boolean areContentsTheSame(int oldItemPosition, int newItemPosition) {
        MyData oldData = oldList.get(oldItemPosition);
        MyData newData = newList.get(newItemPosition);
        return oldData.getName().equals(newData.getName()) && oldData.getAge() == newData.getAge();
    }
}

DiffUtil需要实现四个回调方法,getOldListSize()和getNewListSize()返回旧数据集和新数据集的大小,areItemsTheSame()判断两个数据是否为同一个item,areContentsTheSame()判断同一位置的两条数据是否内容相同。

在数据更新时,我们需要在新数据集合成前使用DiffUtil计算差异,将差异应用到适配器中并更新RecyclerView。

MyDiffCallback diffCallback = new MyDiffCallback(mData, newData);
DiffUtil.DiffResult diffResult = DiffUtil.calculateDiff(diffCallback);
mData = newData;
diffResult.dispatchUpdatesTo(this);

三、使用ItemDecoration美化界面和提升性能

ItemDecoration可以为RecyclerView添加自定义的分割线、颜色、边距等,使界面更加美观,同时也可以提高RecyclerView的性能。我们通常使用RecyclerView.ItemDecoration来创建ItemDecoration。

public class MyItemDecoration extends RecyclerView.ItemDecoration {
    private int mDividerHeight;
    private Paint mPaint;
    public MyItemDecoration(int dividerHeight, int dividerColor) {
        mDividerHeight = dividerHeight;
        mPaint = new Paint();
        mPaint.setColor(dividerColor);
        mPaint.setStyle(Paint.Style.FILL);
    }
    @Override
    public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) {
        outRect.bottom = mDividerHeight;
    }
    @Override
    public void onDraw(Canvas canvas, RecyclerView parent, RecyclerView.State state) {
        int left = parent.getPaddingLeft();
        int right = parent.getWidth() - parent.getPaddingRight();
        for (int i = 0; i < parent.getChildCount() - 1; i++) {
            View child = parent.getChildAt(i);
            float top = child.getBottom();
            canvas.drawRect(left, top, right, top + mDividerHeight, mPaint);
        }
    }
}

在RecyclerView中添加ItemDecoration。

MyItemDecoration itemDecoration = new MyItemDecoration(dp2px(1), Color.GRAY);
recyclerView.addItemDecoration(itemDecoration);

四、使用RecyclerView滑动监听减少内存消耗

RecyclerView在滑动时,会大量的创建和销毁视图,如果在滑动过程中同时处理复杂的数据操作,会导致卡顿或者OOM。因此,我们可以在RecyclerView滑动时停止加载图片或者其他占用内存的操作,从而减少内存消耗,提高滑动性能。我们可以通过RecyclerView的addOnScrollListener()监听RecyclerView的滑动状态。

recyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
    @Override
    public void onScrollStateChanged(@NonNull RecyclerView recyclerView, int newState) {
        super.onScrollStateChanged(recyclerView, newState);
        switch (newState) {
            case RecyclerView.SCROLL_STATE_IDLE:
                //滑动停止,可以执行数据加载等其他操作
                break;
            case RecyclerView.SCROLL_STATE_DRAGGING:
                //手指触摸滑动中,应停止数据加载等流操作
                break;
            case RecyclerView.SCROLL_STATE_SETTLING:
                //惯性滑动中,应停止数据加载等流操作
                break;
        }
    }
});

五、使用LayoutManager优化布局

LayoutManager是RecyclerView中布局管理器的基类,可以用于控制RecyclerView中子View的布局方式。使用LayoutManager可以实现多种复杂的布局,并且可以优化RecyclerView的性能。下面介绍几种常见的LayoutManager。

1. LinearLayoutManager

LinearLayoutManager是RecyclerView默认的布局管理器,控制子View的排列方向,有三种排列方式:VERTICAL、HORIZONTAL、Horizontal.VERTICAL。

LinearLayoutManager layoutManager = new LinearLayoutManager(this);
layoutManager.setOrientation(LinearLayoutManager.VERTICAL);
recyclerView.setLayoutManager(layoutManager);

2. GridLayoutManager

GridLayoutManager可以将子View排列成网格形式。可以通过GridLayoutManager的构造函数来控制列数。

GridLayoutManager layoutManager = new GridLayoutManager(this, 2);
recyclerView.setLayoutManager(layoutManager);

3. StaggeredGridLayoutManager

StaggeredGridLayoutManager可以将子View排列成不规则的网格形式。可以通过StaggeredGridLayoutManager的构造函数来控制列数和方向。

StaggeredGridLayoutManager layoutManager = new StaggeredGridLayoutManager(2, StaggeredGridLayoutManager.VERTICAL);
recyclerView.setLayoutManager(layoutManager);

完整代码示例

public class MainActivity extends AppCompatActivity {
    private RecyclerView recyclerView;
    private List mData;
    private MyAdapter mAdapter;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        recyclerView = findViewById(R.id.recyclerView);
        initData();
        mAdapter = new MyAdapter(mData);
        recyclerView.setAdapter(mAdapter);
        LinearLayoutManager layoutManager = new LinearLayoutManager(this);
        layoutManager.setOrientation(LinearLayoutManager.VERTICAL);
        recyclerView.setLayoutManager(layoutManager);
        MyItemDecoration itemDecoration = new MyItemDecoration(dp2px(1), Color.GRAY);
        recyclerView.addItemDecoration(itemDecoration);
        recyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
            @Override
            public void onScrollStateChanged(@NonNull RecyclerView recyclerView, int newState) {
                super.onScrollStateChanged(recyclerView, newState);
                switch (newState) {
                    case RecyclerView.SCROLL_STATE_IDLE:
                        //滑动停止,可以执行数据加载等其他操作
                        break;
                    case RecyclerView.SCROLL_STATE_DRAGGING:
                        //手指触摸滑动中,应停止数据加载等流操作
                        break;
                    case RecyclerView.SCROLL_STATE_SETTLING:
                        //惯性滑动中,应停止数据加载等流操作
                        break;
                }
            }
        });
    }
    private void initData() {
        mData = new ArrayList();
        mData.add(new MyData(1, "张三", 18));
        mData.add(new MyData(2, "李四", 20));
        mData.add(new MyData(3, "王五", 22));
        mData.add(new MyData(4, "赵六", 21));
    }
    public int dp2px(float dpValue) {
        final float scale = getResources().getDisplayMetrics().density;
        return (int) (dpValue * scale + 0.5f);
    }
    class MyAdapter extends RecyclerView.Adapter {
        private List mData;
        public MyAdapter(List data) {
            mData = data;
        }
        @NonNull
        @Override
        public MyViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
            View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_layout, parent, false);
            MyViewHolder viewHolder = new MyViewHolder(view);
            return viewHolder;
        }
        @Override
        public void onBindViewHolder(MyViewHolder holder, int position) {
            holder.name.setText(mData.get(position).getName());
            holder.age.setText(mData.get(position).getAge());
        }
        @Override
        public int getItemCount() {
            return mData.size();
        }
    }
    class MyViewHolder extends RecyclerView.ViewHolder {
        TextView name;
        TextView age;
        public MyViewHolder(View view) {
            super(view);
            name = (TextView) view.findViewById(R.id.name);
            age = (TextView) view.findViewById(R.id.age);
        }
    }
    class MyData {
        private int id;
        private String name;
        private int age;
        public MyData(int id, String name, int age) {
            this.id = id;
            this.name = name;
            this.age = age;
        }
        public int getId() {
            return id;
        }
        public String getName() {
            return name;
        }
        public int getAge() {
            return age;
        }
    }
    class MyItemDecoration extends RecyclerView.ItemDecoration {
        private int mDividerHeight;
        private Paint mPaint;
        public MyItemDecoration(int dividerHeight, int dividerColor) {
            mDividerHeight = dividerHeight;
            mPaint = new Paint();
            mPaint.setColor(dividerColor);
            mPaint.setStyle(Paint.Style.FILL);
        }
        @Override
        public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) {
            outRect.bottom = mDividerHeight;
        }
        @Override
        public void onDraw(Canvas canvas, RecyclerView parent, RecyclerView.State state) {
            int left = parent.getPaddingLeft();
            int right = parent.getWidth() - parent.getPaddingRight();
            for (int i = 0; i < parent.getChildCount() - 1; i++) {
                View child = parent.getChildAt(i);
                float top = child.getBottom();
                canvas.drawRect(left, top, right, top + mDividerHeight, mPaint);
            }
        }
    }
    class MyDiffCallback extends DiffUtil.Callback {
        private List oldList;
        private List newList;
        public MyDiffCallback(List oldList, List newList) {
            this.oldList = oldList;
            this.newList = newList;
        }
        @Override
        public int getOldListSize() {
            return oldList.size();
        }
        @Override
        public int getNewListSize() {
            return newList.size();
        }
        @Override
        public boolean areItemsTheSame(int oldItemPosition, int newItemPosition) {
            return oldList.get(oldItemPosition).getId() == newList.get(newItemPosition).getId();
        }
        @Override
        public boolean areContentsTheSame(int oldItemPosition, int newItemPosition) {
            MyData oldData = oldList.get(oldItemPosition);
            MyData newData = newList.get(newItemPosition);
            return oldData.getName().equals(newData.getName()) && oldData.getAge() == newData.getAge();
        }
    }
}

原创文章,作者:小蓝,如若转载,请注明出处:https://www.506064.com/n/259389.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
小蓝小蓝
上一篇 2024-12-15 16:28
下一篇 2024-12-15 16:28

相关推荐

  • 使用vscode建立UML图的实践和技巧

    本文将重点介绍在使用vscode在软件开发中如何建立UML图,并且给出操作交互和技巧的指导。 一、概述 在软件开发中,UML图是必不可少的重要工具之一。它为软件架构和各种设计模式的…

    编程 2025-04-29
  • 如何优化 Git 性能和重构

    本文将提供一些有用的提示和技巧来优化 Git 性能并重构代码。Git 是一个非常流行的版本控制系统,但是在处理大型代码仓库时可能会有一些性能问题。如果你正在处理这样的问题,本文将会…

    编程 2025-04-29
  • 优秀周记1000字的撰写思路与技巧

    优秀周记是每个编程开发工程师记录自己工作生活的最佳方式之一。本篇文章将从周记的重要性、撰写思路、撰写技巧以及周记的示例代码等角度进行阐述。 一、周记的重要性 作为一名编程开发工程师…

    编程 2025-04-28
  • 使用@Transactional和分表优化数据交易系统的性能和可靠性

    本文将详细介绍如何使用@Transactional和分表技术来优化数据交易系统的性能和可靠性。 一、@Transactional的作用 @Transactional是Spring框…

    编程 2025-04-28
  • Android ViewPager和ScrollView滑动冲突问题

    Android开发中,ViewPager和ScrollView是两个常用的控件。但是当它们同时使用时,可能会发生滑动冲突的问题。本文将从多个方面介绍解决Android ViewPa…

    编程 2025-04-28
  • Android如何点击其他区域收起软键盘

    在Android应用中,当输入框获取焦点弹出软键盘后,我们希望能够点击其他区域使软键盘消失,以提升用户体验。本篇文章将说明如何实现这一功能。 一、获取焦点并显示软键盘 在Andro…

    编程 2025-04-28
  • Python性能优化方案

    本文将从多个方面介绍Python性能优化方案,并提供相应的示例代码。 一、使用Cython扩展 Cython是一个Python编译器,可以将Python代码转化为C代码,可显著提高…

    编程 2025-04-28
  • Python AUC:模型性能评估的重要指标

    Python AUC是一种用于评估建立机器学习模型性能的重要指标。通过计算ROC曲线下的面积,AUC可以很好地衡量模型对正负样本的区分能力,从而指导模型的调参和选择。 一、AUC的…

    编程 2025-04-28
  • Python性能分析: 如何快速提升Python应用程序性能

    Python是一个简洁高效的编程语言。在大多数情况下,Python的简洁和生产力为开发人员带来了很大便利。然而,针对应用程序的性能问题一直是Python开发人员需要面对的一个难题。…

    编程 2025-04-27
  • 堆叠图配色技巧分享

    堆叠图是数据可视化中常用的一种表现形式,而配色则是影响堆叠图观感和传达信息的重要因素之一。本文将分享一些堆叠图配色的技巧,帮助你创造更好的数据可视化。 一、色彩搭配原则 色彩是我们…

    编程 2025-04-27

发表回复

登录后才能评论