android recyclerview不一致检测到无效的回收站视图位置

时间:2017-05-16 07:19:01

标签: android android-recyclerview

我的应用程序中有回收站视图。用户界面与Google Play商店应用类似。它有两个视图寻呼机以及备用方式的列表和网格。所有数据都从Web服务获取,并在两个API调用中分叉。列表和网格的数据是从另一个API填充的。问题是当我快速滚动recyclerview时,我遇到了这个崩溃。在滚动recyclerview时,用于加载列表/网格中数据的API调用来自bindData()。 阅读有关此主题的众多问题,但无法解决问题。

Fatal Exception: java.lang.IndexOutOfBoundsException: Inconsistency detected. Invalid view holder adapter positionViewHolder{396df09 position=3 id=-1, oldPos=0, pLpos:0 scrap [attachedScrap] tmpDetached no parent}
       at android.support.v7.widget.RecyclerView$Recycler.validateViewHolderForOffsetPosition(RecyclerView.java:4505)
       at android.support.v7.widget.RecyclerView$Recycler.getViewForPosition(RecyclerView.java:4636)
       at android.support.v7.widget.RecyclerView$Recycler.getViewForPosition(RecyclerView.java:4617)
       at android.support.v7.widget.LinearLayoutManager$LayoutState.next(LinearLayoutManager.java:1994)
       at android.support.v7.widget.LinearLayoutManager.layoutChunk(LinearLayoutManager.java:1390)
       at android.support.v7.widget.LinearLayoutManager.fill(LinearLayoutManager.java:1353)
       at android.support.v7.widget.LinearLayoutManager.onLayoutChildren(LinearLayoutManager.java:574)
       at android.support.v7.widget.RecyclerView.dispatchLayoutStep1(RecyclerView.java:2979)
       at android.support.v7.widget.RecyclerView.onMeasure(RecyclerView.java:2619)
       at android.view.View.measure(View.java:18811)
       at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5952)
       at android.widget.FrameLayout.onMeasure(FrameLayout.java:194)
       at android.support.v7.widget.CardView.onMeasure(CardView.java:208)
       at android.view.View.measure(View.java:18811)
       at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5952)
       at android.widget.LinearLayout.measureChildBeforeLayout(LinearLayout.java:1465)
       at android.widget.LinearLayout.measureVertical(LinearLayout.java:748)
       at android.widget.LinearLayout.onMeasure(LinearLayout.java:630)
       at android.view.View.measure(View.java:18811)
       at android.support.v7.widget.RecyclerView$LayoutManager.measureChildWithMargins(RecyclerView.java:7487)
       at android.support.v7.widget.LinearLayoutManager.layoutChunk(LinearLayoutManager.java:1416)
       at android.support.v7.widget.LinearLayoutManager.fill(LinearLayoutManager.java:1353)
       at android.support.v7.widget.LinearLayoutManager.onLayoutChildren(LinearLayoutManager.java:574)
       at android.support.v7.widget.RecyclerView.dispatchLayoutStep2(RecyclerView.java:3028)
       at android.support.v7.widget.RecyclerView.dispatchLayout(RecyclerView.java:2906)
       at android.support.v7.widget.RecyclerView.onLayout(RecyclerView.java:3283)
       at android.view.View.layout(View.java:16653)
       at android.view.ViewGroup.layout(ViewGroup.java:5438)
       at android.widget.FrameLayout.layoutChildren(FrameLayout.java:336)
       at android.widget.FrameLayout.onLayout(FrameLayout.java:273)
       at android.view.View.layout(View.java:16653)
       at android.view.ViewGroup.layout(ViewGroup.java:5438)
       at android.widget.FrameLayout.layoutChildren(FrameLayout.java:336)
       at android.widget.FrameLayout.onLayout(FrameLayout.java:273)
       at android.view.View.layout(View.java:16653)
       at android.view.ViewGroup.layout(ViewGroup.java:5438)
       at android.support.design.widget.HeaderScrollingViewBehavior.layoutChild(HeaderScrollingViewBehavior.java:122)
       at android.support.design.widget.ViewOffsetBehavior.onLayoutChild(ViewOffsetBehavior.java:42)
       at android.support.design.widget.AppBarLayout$ScrollingViewBehavior.onLayoutChild(AppBarLayout.java:1192)
       at android.support.design.widget.CoordinatorLayout.onLayout(CoordinatorLayout.java:814)
       at android.view.View.layout(View.java:16653)
       at android.view.ViewGroup.layout(ViewGroup.java:5438)
       at android.support.v4.widget.DrawerLayout.onLayout(DrawerLayout.java:1187)
       at android.view.View.layout(View.java:16653)
       at android.view.ViewGroup.layout(ViewGroup.java:5438)
       at android.widget.FrameLayout.layoutChildren(FrameLayout.java:336)
       at android.widget.FrameLayout.onLayout(FrameLayout.java:273)
       at android.view.View.layout(View.java:16653)
       at android.view.ViewGroup.layout(ViewGroup.java:5438)
       at android.widget.LinearLayout.setChildFrame(LinearLayout.java:1743)
       at android.widget.LinearLayout.layoutVertical(LinearLayout.java:1586)
       at android.widget.LinearLayout.onLayout(LinearLayout.java:1495)
       at android.view.View.layout(View.java:16653)
       at android.view.ViewGroup.layout(ViewGroup.java:5438)
       at android.widget.FrameLayout.layoutChildren(FrameLayout.java:336)
       at android.widget.FrameLayout.onLayout(FrameLayout.java:273)
       at android.view.View.layout(View.java:16653)
       at android.view.ViewGroup.layout(ViewGroup.java:5438)
       at android.widget.LinearLayout.setChildFrame(LinearLayout.java:1743)
       at android.widget.LinearLayout.layoutVertical(LinearLayout.java:1586)
       at android.widget.LinearLayout.onLayout(LinearLayout.java:1495)
       at android.view.View.layout(View.java:16653)
       at android.view.ViewGroup.layout(ViewGroup.java:5438)
       at android.widget.FrameLayout.layoutChildren(FrameLayout.java:336)
       at android.widget.FrameLayout.onLayout(FrameLayout.java:273)
       at com.android.internal.policy.PhoneWindow$DecorView.onLayout(PhoneWindow.java:2680)
       at android.view.View.layout(View.java:16653)
       at android.view.ViewGroup.layout(ViewGroup.java:5438)
       at android.view.ViewRootImpl.performLayout(ViewRootImpl.java:2198)
       at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:1958)
       at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:1134)
       at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:6045)
       at android.view.Choreographer$CallbackRecord.run(Choreographer.java:860)
       at android.view.Choreographer.doCallbacks(Choreographer.java:672)
       at android.view.Choreographer.doFrame(Choreographer.java:608)
       at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:846)
       at android.os.Handler.handleCallback(Handler.java:739)
       at android.os.Handler.dispatchMessage(Handler.java:95)
       at android.os.Looper.loop(Looper.java:148)
       at android.app.ActivityThread.main(ActivityThread.java:5441)
       at java.lang.reflect.Method.invoke(Method.java)
       at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:738)
       at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:628)

这是我的主要Recyclerview适配器:

public class HomeScreenDataAdapterv2 extends RecyclerView.Adapter<HomeScreenViewHolder> {
    private HomeScreenActionHandler homeScreenActionHandler;
    private ArrayList<HomeScreenParentDataModel> homeScreenParentDataModels;

    public HomeScreenDataAdapterv2(ArrayList<HomeScreenParentDataModel> homeScreenParentDataModels) {
        this.homeScreenParentDataModels = homeScreenParentDataModels;
    }

    @Override
    public HomeScreenViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        View holderView;
        Context ctx = parent.getContext();
        LayoutInflater inflater = LayoutInflater.from(ctx);
        switch (viewType) {
            case HomeScreenDataViewTypes.HOME_SCREEN_MAIN_BANNER_VIEW_TYPE:
                holderView = inflater.inflate(R.layout.home_main_banner_adapter, parent, false);
                return new HomeScreenMainBannerViewHolder(ctx, holderView, homeScreenActionHandler);

            case HomeScreenDataViewTypes.HOME_SCREEN_HOT_DEALS_VIEW_TYPE:
                holderView = inflater.inflate(R.layout.home_hot_deals_adapter, parent, false);
                return new HotDealsViewHolder(ctx, holderView, homeScreenActionHandler);

            case HomeScreenDataViewTypes.HOME_SCREEN_DYNAMIC_BLOCKS_LIST_VIEW_TYPE:
                holderView = inflater.inflate(R.layout.home_screen_dynamic_blocks_parent_list_adapter, parent, false);
                return new HomeScreenDynamicBlocksListViewHolder(ctx, holderView, homeScreenActionHandler);

            case HomeScreenDataViewTypes.HOME_SCREEN_DYNAMIC_BLOCK_GRID_VIEW_TYPE:
                holderView = inflater.inflate(R.layout.home_screen_dynamic_blocks_parent_grid_adapter, parent, false);
                return new HomeScreenDynamicBlocksGridViewHolder(ctx, holderView, homeScreenActionHandler);
            default:
                return null;
        }
    }

    @Override
    public void onBindViewHolder(HomeScreenViewHolder holder, int position) {
        HomeScreenParentDataModel homeScreenParentDataModel = homeScreenParentDataModels.get(position);
        switch (homeScreenParentDataModels.get(position).getHome_screen_view_type()) {
            case HomeScreenDataViewTypes.HOME_SCREEN_MAIN_BANNER_VIEW_TYPE:
                ((HomeScreenMainBannerViewHolder) holder).bindData((HomeBannerDataModel) homeScreenParentDataModel);
                break;

            case HomeScreenDataViewTypes.HOME_SCREEN_DYNAMIC_BLOCKS_LIST_VIEW_TYPE:
                ((HomeScreenDynamicBlocksListViewHolder) holder).bindListData((HomeScreenDynamicBlocksDataModel) homeScreenParentDataModel);
                break;

            case HomeScreenDataViewTypes.HOME_SCREEN_DYNAMIC_BLOCK_GRID_VIEW_TYPE:
                ((HomeScreenDynamicBlocksGridViewHolder) holder).bindGridData((HomeScreenDynamicBlocksDataModel) homeScreenParentDataModel);
                break;
        }
    }

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


    @Override
    public int getItemViewType(int position) {
        return homeScreenParentDataModels.get(position).getHome_screen_view_type();
    }

    private static class HomeScreenMainBannerViewHolder extends HomeScreenViewHolder {
        private Context ctx;
        private HomeScreenActionHandler homeScreenActionHandler;
        private ViewPager main_banner_view_pager;
        private CirclePageIndicator pager_indicator;

        HomeScreenMainBannerViewHolder(Context ctx, View itemView, HomeScreenActionHandler homeScreenActionHandler) {
            super(itemView);
            this.ctx = ctx;
            this.homeScreenActionHandler = homeScreenActionHandler;
            main_banner_view_pager = (ViewPager) itemView.findViewById(R.id.main_banner_view_pager);
            main_banner_view_pager.setClipToPadding(false);


    int padding = Util.convertDptoPixel(ctx, 9.0f);
                main_banner_view_pager.setPageMargin(
Util.convertDptoPixel(ctx, 9.0f));
            main_banner_view_pager.setPadding(padding, 0, padding, 0);
            pager_indicator = (CirclePageIndicator) itemView.findViewById(R.id.pager_indicator);
        }

        public void bindData(HomeBannerDataModel homeMainBannerDataModel) {
            HomeMainBannerPagerAdapter mainBannerAdapter = new HomeMainBannerPagerAdapter(ctx, homeMainBannerDataModel.getBannerList());
            mainBannerAdapter.setHomeScreenActionHandler(homeScreenActionHandler);
            main_banner_view_pager.setAdapter(mainBannerAdapter);
            pager_indicator.setViewPager(main_banner_view_pager);
        }
    }

    private static class HomeScreenDynamicBlocksListViewHolder extends HomeScreenViewHolder implements View.OnClickListener {
        private HomeScreenActionHandler homeScreenActionHandler;
        private RubikMediumTextView section_name;
        private RubikMediumButton btn_view_all;
        private RecyclerView dynamic_widgets_container;
        private Widget widget;
        private HomeScreenBlockListAdapter blockListAdapter;

        HomeScreenDynamicBlocksListViewHolder(Context ctx, View itemView, HomeScreenActionHandler homeScreenActionHandler) {
            super(itemView);
            this.homeScreenActionHandler = homeScreenActionHandler;
            section_name = (RubikMediumTextView) itemView.findViewById(R.id.section_name);
            btn_view_all = (RubikMediumButton) itemView.findViewById(R.id.btn_view_all);
            btn_view_all.setOnClickListener(this);

            dynamic_widgets_container = (RecyclerView) itemView.findViewById(R.id.dynamic_widgets_container);
            section_name.setTextColor(Color.parseColor("#546682"));
            LinearLayoutManager linearLayoutManager = new LinearLayoutManager(ctx);
            linearLayoutManager.setOrientation(LinearLayoutManager.VERTICAL);
            linearLayoutManager.setAutoMeasureEnabled(true);
            dynamic_widgets_container.setLayoutManager(linearLayoutManager);
            dynamic_widgets_container.addItemDecoration(new DividerItemDecoration(ctx));
            dynamic_widgets_container.setNestedScrollingEnabled(false);
        }

        void bindListData(HomeScreenDynamicBlocksDataModel dynamicBlocksDataModel) {
            widget = dynamicBlocksDataModel.getWidget();
            if (widget != null) {
                section_name.setText(widget.getName());
                ArrayList<BuyListingsModel> dynamicModels = widget.getDynamicDataModels();
                if (blockListAdapter == null) {
                    blockListAdapter = new HomeScreenBlockListAdapter(dynamicModels, homeScreenActionHandler);
                    dynamic_widgets_container.setAdapter(blockListAdapter);
                }
                if (dynamicModels.isEmpty()) {
                    homeScreenActionHandler.fetchWidgetData(widget.getName(), getAdapterPosition(), dynamicBlocksDataModel, blockListAdapter);
                }
            }
        }

        @Override
        public void onClick(View v) {
            if (homeScreenActionHandler != null && widget != null) {
                homeScreenActionHandler.gotoSearchResult(widget.getListingFilters());
            }
        }
    }

    private static class HomeScreenDynamicBlocksGridViewHolder extends HomeScreenViewHolder implements View.OnClickListener {
        private HomeScreenActionHandler homeScreenActionHandler;
        private RubikMediumTextView section_name;
        private RubikMediumButton btn_view_all;
        private RecyclerView dynamic_widgets_container;
        private Widget widget;
        private HomeScreenBlockGridAdapter blockGridAdapter;

        HomeScreenDynamicBlocksGridViewHolder(Context ctx, View itemView, HomeScreenActionHandler homeScreenActionHandler) {
            super(itemView);
            this.homeScreenActionHandler = homeScreenActionHandler;
            section_name = (RubikMediumTextView) itemView.findViewById(R.id.section_name);
            btn_view_all = (RubikMediumButton) itemView.findViewById(R.id.btn_view_all);
            btn_view_all.setOnClickListener(this);

            dynamic_widgets_container = (RecyclerView) itemView.findViewById(R.id.dynamic_widgets_container);
            GridLayoutManager gridLayoutManager = new GridLayoutManager(ctx, 2);
            gridLayoutManager.setAutoMeasureEnabled(true);
            dynamic_widgets_container.addItemDecoration(new GridSpacingItemDecoration(2, 1, true));
            dynamic_widgets_container.setNestedScrollingEnabled(false);
            dynamic_widgets_container.setLayoutManager(gridLayoutManager);
        }

        void bindGridData(HomeScreenDynamicBlocksDataModel dynamicBlocksDataModel) {
            widget = dynamicBlocksDataModel.getWidget();
            if (widget != null) {
                section_name.setText(widget.getName());
                ArrayList<BuyListingsModel> dynamicModels = widget.getDynamicDataModels();
                if (blockGridAdapter == null) {
                    blockGridAdapter = new HomeScreenBlockGridAdapter(dynamicModels, homeScreenActionHandler);
                    dynamic_widgets_container.setAdapter(blockGridAdapter);
                }

                if (dynamicModels.isEmpty()) {
                    homeScreenActionHandler.fetchWidgetData(widget.getName(), getAdapterPosition(), dynamicBlocksDataModel, blockGridAdapter);
                }
            }
        }

        @Override
        public void onClick(View v) {
            if (homeScreenActionHandler != null && widget != null) {
                homeScreenActionHandler.gotoSearchResult(widget.getListingFilters());
            }
        }
    }

    public interface HomeScreenActionHandler {
        void gotoCategoryListing(String categoryName);

        void gotoCategoryPage();

        void handleBannerClick(Banner banner);

        void gotoSearchResult(Bundle listingFiltersBundle);

        void fetchWidgetData(String widgetName, int adapterPosition, HomeScreenDynamicBlocksDataModel dynamicBlocksDataModel, RecyclerView.Adapter listAdapter);

        void performWishlistAction(BuyListingsModel buyListingsModel, int childApdaterPosition, RecyclerView.Adapter listAdapter);

        void performBlockClick(String listing_id, boolean isInWatchlist);
    }

    public void setHomeScreenActionHandler(HomeScreenActionHandler homeScreenActionHandler) {
        this.homeScreenActionHandler = homeScreenActionHandler;
    }
}

这是用于在列表中加载数据的API调用:

public void fetchWidgetData(final String widgetName, final int adapterPosition,
                                final HomeScreenDynamicBlocksDataModel dynamicBlocksDataModel, final RecyclerView.Adapter listAdapter) {
        final Widget widget = dynamicBlocksDataModel.getWidget();

        Bundle b = widget.getListingFilters();
        HashMap<String, String> params = new HashMap<>();
        if (!b.containsKey("recently_sold"))
            params.put("isSearch", "1");
        Set<String> keySet = b.keySet();
        Iterator<String> it = keySet.iterator();
        while (it.hasNext()) {
            String key = it.next();
            params.put(key, b.getString(key));
        }
        Response.Listener<JSONObject> responseListener = new Response.Listener<JSONObject>() {

            @Override
            public void onResponse(JSONObject response) {
                                    try {
                    String code = response.getString("code");
                    if (code.equalsIgnoreCase("success")) {
                        JSONArray dataArray = response.getJSONArray("data");
                        ArrayList<BuyListingsModel> models = HomeScreenDataParser
                                .getDynamicBlocksModelList(dataArray);
                        if (models != null && !models.isEmpty()) {
                            ArrayList<BuyListingsModel> buyListingsModels = widget.getDynamicDataModels();
                            buyListingsModels.clear();
                            if (listAdapter != null) {
                                int size = models.size();
                                int upperbound = listAdapter instanceof HomeScreenBlockListAdapter ? 3 : 4;
                                for (int i = 0; i < size && i < upperbound; i++) {
                                    buyListingsModels.add(models.get(i));
                                }
                                listAdapter.notifyItemRangeInserted(0, buyListingsModels.size());

                            }
                            //homeScreenDataAdapterv2.notifyItemChanged(adapterPosition);
                        }
                    } else if (code.equalsIgnoreCase("failed")) {
                        handleError(response);
                    }
                } catch (JSONException e) {
                    e.printStackTrace();
                }
            }
        };

        Response.ErrorListener errorListener = new Response.ErrorListener() {

            @Override
            public void onErrorResponse(VolleyError error) {
                error.printStackTrace();
            }
        };
        Api.getHomeScreenWidgetDetails(widgetName, responseListener, errorListener, ctx, params);

3 个答案:

答案 0 :(得分:15)

由于您实际上正在替换所有数据,因此调用notifyItemRangeInserted()而不实际向适配器插入任何内容会引发此问题。您必须在将实际数据设置到适配器后调用正确的通知程序,或者更准确地将列表列入基础列表。

notifyItemRangeInserted(pos, count)通知适配器期望在count插入pos个项目。因此,列表的大小应该增加count,而不仅仅是count。由于您替换了所有数据,因此应改为调用notifyDataSetChanged()或使用DiffUtil来计算数据的实际差异。但是,您不必担心再使用哪个通知。

listAdapter.notifyDataSetChanged();

答案 1 :(得分:5)

我有同样的问题,我在互联网上搜索并尝试了很多解决方案。最后,我的问题通过将RecyclerView的ItemAnimator设置为null来解决。

    Rails.application.routes.draw do
        root "users#index"
        get '/' => "users#index"
        get "/new1" => "users#new1"
        get "/new2" => "users#new2"
        get "/new3" => "users#new3"
        resources :users ,to: 'posts'
        post "/users" => "users#create"
        get "users/:id" => "users#show"
        get    '/login',   to: 'sessions#new'
        post   '/login',   to: 'sessions#create'
        get '/logout',  to: 'sessions#destroy'
        delete '/logout',  to: 'sessions#destroy'
        resources :sessions
        resources :posts
        get "users/:id/post" => "posts#index"
     end

这将解决问题

答案 2 :(得分:0)

在onBindViewHolder(@NonNull RecyclerView.ViewHolder支架,int位置)检查中,

if(position != RecyclerView.NO_POSITION){
   // Do your binding here
}

这对我有用。