2016-02-23 17 views
5

Mam ten ślad stosu mimo crashlytics. Nie mam pojęcia, gdzie jest problem.Recycler Wyświetl wyjątek krytyczny: java.lang.ArrayIndexOutOfBoundsException

Czy istnieje alternatywa dla StaggeredGridLayoutManager, której można użyć do uzyskania układu podobnego do listy.

Fatal Exception: java.lang.ArrayIndexOutOfBoundsException: start < 0 || end > len. start=-1, end=11, len=11 
    at java.util.Arrays.checkStartAndEnd(Arrays.java:1745) 
    at java.util.Arrays.fill(Arrays.java:803) 
    at android.support.v7.widget.StaggeredGridLayoutManager$LazySpanLookup.invalidateAfter(SourceFile:2404) 
    at android.support.v7.widget.StaggeredGridLayoutManager.handleUpdate(SourceFile:1373) 
    at android.support.v7.widget.StaggeredGridLayoutManager.onItemsRemoved(SourceFile:1327) 
    at android.support.v7.widget.RecyclerView$6.dispatchUpdate(SourceFile:725) 
    at android.support.v7.widget.RecyclerView$6.onDispatchFirstPass(SourceFile:716) 
    at android.support.v7.widget.AdapterHelper.dispatchFirstPassAndUpdateViewHolders(SourceFile:316) 
    at android.support.v7.widget.AdapterHelper.dispatchAndUpdateViewHolders(SourceFile:302) 
    at android.support.v7.widget.AdapterHelper.applyRemove(SourceFile:182) 
    at android.support.v7.widget.AdapterHelper.preProcess(SourceFile:103) 
    at android.support.v7.widget.RecyclerView.processAdapterUpdatesAndSetAnimationFlags(SourceFile:2767) 
    at android.support.v7.widget.RecyclerView.onMeasure(SourceFile:2538) 
    at android.view.View.measure(View.java:16521) 
    at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5134) 
    at android.widget.LinearLayout.measureChildBeforeLayout(LinearLayout.java:1404) 
    at android.widget.LinearLayout.measureVertical(LinearLayout.java:695) 
    at android.widget.LinearLayout.onMeasure(LinearLayout.java:588) 
    at android.view.View.measure(View.java:16521) 
    at android.support.v4.view.ViewPager.onMeasure(SourceFile:1489) 
    at android.view.View.measure(View.java:16521) 
    at android.widget.RelativeLayout.measureChildHorizontal(RelativeLayout.java:719) 
    at android.widget.RelativeLayout.onMeasure(RelativeLayout.java:455) 
    at android.view.View.measure(View.java:16521) 
    at android.widget.RelativeLayout.measureChildHorizontal(RelativeLayout.java:719) 
    at android.widget.RelativeLayout.onMeasure(RelativeLayout.java:455) 
    at android.view.View.measure(View.java:16521) 
    at android.support.v4.widget.DrawerLayout.onMeasure(SourceFile:940) 
    at android.view.View.measure(View.java:16521) 
    at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5134) 
    at android.widget.FrameLayout.onMeasure(FrameLayout.java:310) 
    at android.support.v7.widget.ContentFrameLayout.onMeasure(SourceFile:135) 
    at android.view.View.measure(View.java:16521) 
    at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5134) 
    at android.widget.FrameLayout.onMeasure(FrameLayout.java:310) 
    at android.view.View.measure(View.java:16521) 
    at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5134) 
    at android.widget.FrameLayout.onMeasure(FrameLayout.java:310) 
    at android.view.View.measure(View.java:16521) 
    at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5134) 
    at android.widget.LinearLayout.measureChildBeforeLayout(LinearLayout.java:1404) 
    at android.widget.LinearLayout.measureVertical(LinearLayout.java:695) 
    at android.widget.LinearLayout.onMeasure(LinearLayout.java:588) 
    at android.view.View.measure(View.java:16521) 
    at android.view.ViewGroup.measureChildWithMargins(ViewGroup.java:5134) 
    at android.widget.FrameLayout.onMeasure(FrameLayout.java:310) 
    at com.android.internal.policy.impl.PhoneWindow$DecorView.onMeasure(PhoneWindow.java:2331) 
    at android.view.View.measure(View.java:16521) 
    at android.view.ViewRootImpl.performMeasure(ViewRootImpl.java:1925) 
    at android.view.ViewRootImpl.measureHierarchy(ViewRootImpl.java:1117) 
    at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:1299) 
    at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:1004) 
    at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:5739) 
    at android.view.Choreographer$CallbackRecord.run(Choreographer.java:761) 
    at android.view.Choreographer.doCallbacks(Choreographer.java:574) 
    at android.view.Choreographer.doFrame(Choreographer.java:544) 
    at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:747) 
    at android.os.Handler.handleCallback(Handler.java:733) 
    at android.os.Handler.dispatchMessage(Handler.java:95) 
    at android.os.Looper.loop(Looper.java:136) 
    at android.app.ActivityThread.main(ActivityThread.java:5380) 
    at java.lang.reflect.Method.invokeNative(Method.java) 
    at java.lang.reflect.Method.invoke(Method.java:515) 
    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:970) 
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:786) 
    at dalvik.system.NativeStart.main(NativeStart.java) 

To co robie obsłużyć kliknij słuchaczy w jednym z adapterem

 @Override 
public void onBindViewHolder(final ViewHolder holder, final int position) { 
    holder.relative.setClickable(true); 

    holder.relative.setOnClickListener(new View.OnClickListener() { 

     @Override 
     public void onClick(View v) { 
      if (itemClickListener != null) 
       itemClickListener.onClick(v, position); 
      else 
       AppConfig.log("itemClickLestener was null"); 
     } 
    }); 

    holder.relative.setOnLongClickListener(new View.OnLongClickListener() { 
     @Override 
     public boolean onLongClick(View v) { 
      if (itemClickListener != null) 
       itemClickListener.onLongClick(v, position); 
      return true; 
     } 
    }); 
    return view; 
    } 

Odpowiedz

15

Wygląda na to, że próbujesz wywołać getAdapterPosition(), podczas gdy układy widoku Recycler View nie są obliczane (prawdopodobnie wywoływane przez wywołanie notifyDataSetChanged()).

zawsze poradzić jak to

final int position = getAdapterPosition(); 
if(positon != RecyclerView.NO_POSITION){ 
    // you can trust the adapter position 
    // do whatever you intend to do with this position 
} 

Dokumentacja jasno stwierdza co następuje

jeśli chcesz uzyskać dostęp do pozycji w adapterze na ViewHolder kliknięcia, należy użyć getAdapterPosition(). Należy pamiętać, że metody te mogą nie być w stanie obliczyć pozycji adaptera, jeśli wywołano funkcję notifyDataSetChanged(), a nowy układ nie został jeszcze obliczony. Z tego powodu należy ostrożnie obsługiwać NO_POSITION lub zerowe wyniki z tych metod.

UPDATE

Teraz, gdy w końcu pisał jakiś kod. Oto kilka rzeczy, które powinieneś rozważyć.

  1. Nie należy przekazywać końcowych parametrów w obiekcie onBindViewHolder. Do czasu uruchomienia OnClickListener pozycja nie może być taka sama, jak by się wydawało. Więc to zrobić

    @Override 
    public void onBindViewHolder(ViewHolder holder, int position) { 
        holder.relative.setClickable(true); 
        //do other stuff 
    } 
    
  2. NIE przypisać jakiekolwiek detektory wewnątrz metody onBindViewHolder. Gdy widok zostanie poddany recyklingowi, ustawiamy detektory kilka razy w tym samym widoku. Zamiast tego, należy ustawić go wewnątrz klasy widok posiadacz lub onCreateViewHolder metody jak poniżej

    @Override 
    public MyViewHolder onCreateViewHolder (ViewGroup parent, int viewType) { 
        View myView = LayoutInflater.from(parent.getContext()) 
             .inflate(R.layout.my_layout, parent, false); 
    
        //setting listeners on creation 
        myView.setOnClickListener(new View.OnClickListener(){ 
         @Override 
         public void onClick(View v) { 
          final int position = getAdapterPosition(); 
          if(position != RecyclerView.NO_POSITION){ 
           //do whatever you want with the clicked position 
          } 
         } 
        }); 
    
        //return the view holder 
        return new MyViewHolder(myView); 
    } 
    

    Albo w klasie ViewHolder

    public class MyViewHolder extends RecyclerView.ViewHolder { 
    
        public MyViewHolder(View itemView) { 
         super(itemView); 
    
         //setting listeners inside viewHolder class 
         itemView.setOnClickListener(new View.OnClickListener(){ 
          @Override 
          public void onClick(View v) { 
           final int position = getAdapterPosition(); 
           if(position != RecyclerView.NO_POSITION){ 
            //do whatever you want with the clicked position 
           } 
          } 
          }); 
         } 
        } 
    

Mocowanie powyżej może rozwiązać Twój problem. Gorąco polecam zapoznać się z nagraniem Google Dev Summit 2015 na temat RecyclerView. To jest naprawdę interesujące i mówią o tajnikach RecyclerView. Wspominają także, dlaczego nie przekazywać parametrów końcowych w onBinViewHolder. Kliknij here, aby sprawdzić dokładny moment w filmie, w którym o nim mówią.

+0

Mam mój kod teraz przy użyciu nowego GridLayoutManager (this.getActivity(), 1) co myślisz o tym, nie znalazłem żadnego kodu przy użyciu getAdapterPosition ( –

+0

Nie sądzę, że ma coś wspólnego z Menedżera układu, którego używasz.Możesz opublikować swoją klasę adaptera? –

+0

Użyłem StaggeredGridLayoutManager w trzech Adapters.Nie wiem, gdzie dokładnie ten problem wynika z. Więc nie można udostępnić żadnego szczegółu –

0

Głównie ten błąd zostanie występującego w swojej sprawie BindView gdzie będzie używana dane swojej ArrayList. Upewnij się, że użyłeś wartości pozycji -1 wartości z listy tablic.

+0

Mam mój kod teraz przy użyciu nowego GridLayoutManager (this.getActivity(), 1) co myślisz o tym –

+0

@RajnishMishra jest w porządku, ale czy możesz po prostu pokazać nam bindviewholder? –

+0

Zaktualizowałem mój Ques –

Powiązane problemy