我有一个大约5-6行的ListView,每一行都是一个显示水平片段的RecycleView。卡片(例如Netflix,HBO应用程序)
考虑:
我为ListView适配器中的RecycleViews创建了一个缓存(SparseArray),以保持每个RecycleView的状态,否则每次它离开屏幕时都会重新启动到位置0。
通过这样做,只需将每个RecycleView滚动到最后,使其中一些得到冻结。
尝试了几种解决方法:
但是,这些选项都没有起作用,保证应用程序没有冻结的唯一方法是避免使用ListView缓存(如果将View移出屏幕,则会丢失您访问过的最后位置)< / p>
Glide Code(我在这里尝试了几个选项:避免使用Bitmap,避免内存缓存,将磁盘策略更新为ALL)
Glide.with(context).load(url)
.asBitmap()
.skipMemoryCache(true)
.thumbnail(0.5f)
.centerCrop()
.error(placeHolderId)
.diskCacheStrategy(DiskCacheStrategy.RESULT)
//.priority(Priority.NORMAL)
.dontAnimate()
.into(new SimpleTarget<Bitmap>() {
@Override
public void onResourceReady(Bitmap bitmap, GlideAnimation anim) {
if (bitmap != null) {
imgView.setImageBitmap(bitmap);
//cache.put(url, bitmap);
} else {
imgView.setBackground(null);
}
}
});
有任何建议/意见吗? 也许还有另一个选项来保持View的状态而没有缓存?
ListView适配器代码:
@Override
public View getView(int position, View convertView, ViewGroup parent) {
case TYPE_VENUE_FILTER: {
View v;
// Create a new view into the list.
LayoutInflater inflater = (LayoutInflater) context
.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
if (inflater != null) {
v = inflater.inflate(R.layout.fragment_venue_filter, parent, false);
}
// We need to get the exact filter to be displayed
int index = position - FIRST_VENUE_FILTER_ROW;
if (index < this.filters.size()) {
VenueFilter filter = this.filters.get(index);
// Recalculate the distance order
filter.sortVenuesForLocation(this.lastLocation);
// Set data into the view.
VenueFilterFragment.loadFragment(this.context, v, filter, this.fm);
}
return v;
}
}
更新11/04
我终于明白了。
此代码位于适配器中,用于对特定行(getView)进行充气
// We should reset or recover the scroll state
LinearLayoutManager llm = (LinearLayoutManager) filterContent.getLayoutManager();
if (filter.offset > 0) {
if (llm != null) llm.scrollToPosition(filter.offset + 1);
} else {
if (llm != null) llm.scrollToPosition(0);
}
// We need to clear every time. Because we are reusing the view
recycleView.clearOnScrollListeners();
// We need to create a new specific scroll listener
recycleView.addOnScrollListener(new RecyclerView.OnScrollListener() {
@Override
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
super.onScrolled(recyclerView, dx, dy);
// Keeping the offset for a future load
LinearLayoutManager llm = (LinearLayoutManager) recyclerView.getLayoutManager();
if (llm != null) filter.offset = llm.findFirstVisibleItemPosition();
}
});
答案 0 :(得分:0)
这是一个非常广泛而复杂的问题。您的布局很复杂,因此需要一些努力才能使其高效。您可以采取哪些措施来改善绩效:
RecyclerView
应尽可能使用视图持有者。如果需要,定义多个视图类型。父RecyclerView
可能只有一种视图类型,即具有子RecyclerView
的项。通过拥有RecyclerView
,子RecycledViewPool
的可能也可以与观看者共享。更多信息:https://developer.android.com/reference/android/support/v7/widget/RecyclerView.RecycledViewPool.html ImageView
状态以避免闪烁。 我希望有所帮助。祝好运!