【问题标题】:RecyclerView creating and binding all ViewHolders when scrollingRecyclerView在滚动时创建和绑定所有ViewHolders
【发布时间】:2016-07-17 21:14:06
【问题描述】:

我有一个无法确定的奇怪问题。我的应用程序从网站获取电影数据,特别是海报图像。我创建了一个 RecyclerView,它应该在 2 列网格中显示所有海报。

滚动体验非常滞后,并且由于某种原因,即使数据列表没有变化,项目也会在滚动中交换位置。

我尝试调试应用程序,发现在我第一次调用 notifyDataSetChanged() 时,我的适配器会为列表中的 每个 项目调用 onCreateViewHolder(...)(大约 90 次,当在任何给定时刻只有 6 张海报可见时),同样适用于 onBindViewAdapter(...)。 然后,在滚动时,它会重新创建并重新绑定所有 ViewHolders!

我一遍又一遍地检查我的代码,但无法确定原因。非常感谢您的帮助。

RecyclerView.adapter:

public class MovieRecyclerViewAdapter extends RecyclerView.Adapter<MovieRecyclerViewAdapter.ViewHolder> {
    private final Cinema cinema;
    private final List<Movie> movies;
    private final Context context;
    private final DisplayMetrics displayMetrics;
    private final Picasso picasso;

    public MovieRecyclerViewAdapter(Context context, Cinema cinema, List<Movie> movies) {
        this.context = context;
        this.cinema = cinema;
        this.movies = movies;

        WindowManager wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
        displayMetrics = new DisplayMetrics();
        wm.getDefaultDisplay().getMetrics(displayMetrics);

        // Set up Picasso with okHttp3
        okhttp3.OkHttpClient okHttp3Client = new okhttp3.OkHttpClient();
        OkHttp3Downloader okHttp3Downloader = new OkHttp3Downloader(okHttp3Client);
        picasso = new Picasso.Builder(context)
                .downloader(okHttp3Downloader)
                .build();
    }

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

    @Override
    public void onBindViewHolder(final ViewHolder holder, int position) {
        Movie movie = movies.get(position);
        int imgViewWidth = displayMetrics.widthPixels / 2;
        int imgViewHeight = (int)(imgViewWidth * 1.4);
        picasso.with(context)
                .load(cinema.getPosterUrl(movie))
                .resize(imgViewWidth, imgViewHeight)
                .into(holder.imageView);
    }

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

    public class ViewHolder extends RecyclerView.ViewHolder {
        public ImageView imageView;

        public ViewHolder(View view) {
            super(view);
            this.imageView = (ImageView) view.findViewById(R.id.fragment_movies_poster);
        }
    }
}

RecyclerView 项目布局:

<?xml version="1.0" encoding="utf-8"?>
<ImageView xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/fragment_movies_poster"
    android:layout_width="match_parent"
    android:layout_height="wrap_content">

</ImageView>

RecyclerView XML 定义:

<android.support.v7.widget.RecyclerView
        android:id="@+id/fragment_movies_recycler_view"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        tools:listitem="@layout/fragment_movies" />

RecyclerView 配置,设置适配器:

@Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.fragment_movies_grid, container, false);

        Context context = view.getContext();
        RecyclerView recyclerView = (RecyclerView) view.findViewById(R.id.fragment_movies_recycler_view);
        recyclerView.setLayoutManager(new GridLayoutManager(context, columnCount));
        movies = new ArrayList<>();
        adapter = new MovieRecyclerViewAdapter(context, cinema, movies);
        recyclerView.setAdapter(adapter);
        recyclerView.setHasFixedSize(true);

        fetchMovies(view); // Populates movies list

        return view;
    }

编辑

自从发布这个问题后,我发现了一个类似的问题:Android RecyclerView Creates and Binds all the views on Dataset change

确实,一旦我将项目 layout_height 设置为固定值而不是“wrap_content”,问题就解决了。显然 RecyclerView 试图创建它的所有视图以确定它的高度。

但是,我尝试了该问题中提供的解决方案,调用 LayoutManager 的 lm.setAutoMeasureEnabled(false) 并将 layout_height 设置为“wrap_content”,问题仍然存在。是否有任何建议我将如何在不发生此问题的情况下具有自适应 layout_height?

【问题讨论】:

    标签: android android-recyclerview


    【解决方案1】:

    我最终通过为项目使用自定义 ImageView 子类来避免此问题,覆盖 onMeasure() 并为其提供根据宽度确定高度的固定纵横比。


    FixedAspectRatioImageView:

    public class FixedAspectRatioImageView extends ImageView{
        public FixedAspectRatioImageView(Context context) {
            super(context);
        }
    
        public FixedAspectRatioImageView(Context context, AttributeSet attrs) {
            super(context, attrs);
        }
    
        public FixedAspectRatioImageView(Context context, AttributeSet attrs, int defStyleAttr) {
            super(context, attrs, defStyleAttr);
        }
    
        @Override
        protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
            int widthSize = MeasureSpec.getSize(widthMeasureSpec);
            int heightSize = (int) (widthSize * 1.4);
            super.onMeasure(widthMeasureSpec, MeasureSpec.makeMeasureSpec(heightSize, MeasureSpec.EXACTLY));
        }
    }
    

    XML 中定义的 RecyclerView 项:

    <?xml version="1.0" encoding="utf-8"?>
    <com.michaelsvit.kolnoa.FixedAspectRatioImageView
        xmlns:android="http://schemas.android.com/apk/res/android"
        android:id="@+id/fragment_movies_poster"
        android:layout_width="match_parent"
        android:layout_height="288dp"> // Value here does not matter, as it is dynamically set
                                       // to be 1.4 times width
    
    </com.michaelsvit.kolnoa.FixedAspectRatioImageView>
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2017-06-21
      • 2021-07-25
      • 1970-01-01
      • 2016-07-24
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多