2016-02-23 16 views
5

Ho questa traccia dello stack nonostante crashlytics. Non ho idea di dove sia il problema.Recycler Visualizza eccezione irreversibile: java.lang.ArrayIndexOutOfBoundsException

Esiste un'alternativa di StaggeredGridLayoutManager che è possibile utilizzare per ottenere un layout simile a una listview.

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) 

Questo è quello che sto facendo per gestire cliccare ascoltatori in uno dell'adattatore

 @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; 
    } 

risposta

15

Sembra che si stava tentando di chiamare getAdapterPosition() mentre Recycler Visualizza i layout non vengono calcolati (eventualmente derivanti da una chiamata notifyDataSetChanged()).

gestire sempre in questo modo

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

La documentazione afferma chiaramente il seguente

se si desidera accedere alla voce l'adattatore su un ViewHolder click, è necessario utilizzare getAdapterPosition(). Fare attenzione che questi metodi potrebbero non essere in grado di calcolare le posizioni dell'adattatore se è stato chiamato notifyDataSetChanged() e il nuovo layout non è stato ancora calcolato. Per questo motivo, dovresti gestire attentamente NO_POSITION o risultati nulli da questi metodi.

UPDATE

Ora che avete finalmente pubblicato po 'di codice. ecco alcune cose che dovresti prendere in considerazione.

  1. Non passare i parametri finali all'interno di BindViewHolder. Nel momento in cui viene attivato OnClickListener, la posizione non è garantita per essere la stessa che si pensa che sarebbe. Quindi fare questo

    @Override 
    public void onBindViewHolder(ViewHolder holder, int position) { 
        holder.relative.setClickable(true); 
        //do other stuff 
    } 
    
  2. Non assegnare eventuali ascoltatori all'interno metodo onBindViewHolder. Quando la vista viene riciclata, si finisce per impostare i listener più volte sulla stessa vista. Invece, impostarlo all'interno della classe di visualizzazione titolare o di un metodo onCreateViewHolder come qui di seguito

    @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); 
    } 
    

    O nella classe 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 
           } 
          } 
          }); 
         } 
        } 
    

Fissaggio quanto sopra può risolvere il problema. Consiglio vivamente di controllare il video di Google Dev Summit 2015 su RecyclerView. È davvero interessante e parlano dei dettagli di RecyclerView. Inoltre menzionano perché non passare i parametri finali all'interno di onBinViewHolder. Fai clic su here per controllare l'esatto momento nel video in cui ne parlano.

+0

Ora ho il mio codice usando il nuovo GridLayoutManager (this.getActivity(), 1) cosa ne pensi, non ho trovato alcun codice usando getAdapterPosition ( –

+0

Non penso È possibile inserire la classe dell'adattatore? –

+0

Stavo usando StaggeredGridLayoutManager in tre adattatori Non so da dove provengano esattamente questi problemi. Quindi non sono in grado di condividere alcun dettaglio –

0

Per lo più questo errore verrà accadendo nella vostra su Bindview in cui si prevede di utilizzare i dati del tuo arraylist. Assicurati di aver utilizzato il valore posizione -1 dell'array.

+0

Ora ho il mio codice utilizzando il nuovo GridLayoutManager (questo.getActivity(), 1) cosa ne pensi di questo –

+0

@RajnishMishra questo va bene ma puoi semplicemente mostrarci il bindviewholder? –

+0

Ho aggiornato le mie Ques –