2013-03-27 167 views
2

爲了理智,我在這裏張貼這個,因爲我確信我現在無法自己解決它。我發佈了所有需要了解的內容,因爲我想爲什麼過去沒有解決這個問題是因爲我沒有發佈所有內容。到目前爲止,我已經剝離了代碼,基本上它是儘可能輕的。但仍然出現此錯誤(儘管非常非常非常罕見):ListView過濾器:IllegalStateException

03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): The content of the adapter has changed but ListView did not receive a notification. Make sure the content of your adapter is not modified from a background thread, but only from the UI thread. [in ListView(16908298, class android.widget.ListView) with Adapter(class de.innosoft.android.mobileserviceclient.activities.ListEinsaetze$ArrayAdapterEinsatz)] 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): java.lang.IllegalStateException: The content of the adapter has changed but ListView did not receive a notification. Make sure the content of your adapter is not modified from a background thread, but only from the UI thread. [in ListView(16908298, class android.widget.ListView) with Adapter(class de.innosoft.android.mobileserviceclient.activities.ListEinsaetze$ArrayAdapterEinsatz)] 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at android.widget.ListView.layoutChildren(ListView.java:1544) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at android.widget.AbsListView.onLayout(AbsListView.java:1994) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at android.widget.AdapterView.updateEmptyStatus(AdapterView.java:747) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at android.widget.AdapterView.checkFocus(AdapterView.java:720) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at android.widget.AdapterView$AdapterDataSetObserver.onChanged(AdapterView.java:812) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at android.widget.AbsListView$AdapterDataSetObserver.onChanged(AbsListView.java:5958) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at android.database.DataSetObservable.notifyChanged(DataSetObservable.java:37) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at android.widget.BaseAdapter.notifyDataSetChanged(BaseAdapter.java:50) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at de.innosoft.android.mobileserviceclient.ui.DynamicArrayAdapter.notifyDataSetChanged(DynamicArrayAdapter.java:291) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at de.innosoft.android.mobileserviceclient.ui.DynamicArrayAdapter.notifyDataSetChanged(DynamicArrayAdapter.java:268) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at de.innosoft.android.mobileserviceclient.ui.DynamicArrayAdapter$DynamicAdapterFilter.publishResults(DynamicArrayAdapter.java:119) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at android.widget.Filter$ResultsHandler.handleMessage(Filter.java:282) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at android.os.Handler.dispatchMessage(Handler.java:99) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at android.os.Looper.loop(Looper.java:137) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at android.app.ActivityThread.main(ActivityThread.java:5039) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at java.lang.reflect.Method.invokeNative(Native Method) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at java.lang.reflect.Method.invoke(Method.java:511) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:793) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:560) 
03-27 10:57:27.783: E/java.lang.IllegalStateException(4929): at dalvik.system.NativeStart.main(Native Method) 

在這篇文章的最底部我解釋出現此錯誤時。但首先,代碼。我有這個TextWatcher:

edSearch.addTextChangedListener(new TextWatcher() { 
       @Override 
       public void onTextChanged(CharSequence s, int start, int before, int count) { 
       } 

       String text = null; 

       @Override 
       public void beforeTextChanged(CharSequence s, int start, int count, int after) { 
        text = s.toString(); 
       } 

       @Override 
       public void afterTextChanged(final Editable s) { 
        if (text != null && !text.equals(s.toString())) { 
         adapter.getFilter().filter(s.toString()); 
        } 
        text = null; 
       } 
      }); 

這個線程在邊線上運行,每200毫秒觸發一次textwatcher。我一定是瘋了手動測試:

new Thread(new Runnable() { 
      public void run() { 
       Runnable r = new Runnable() { 
        public void run() { 
         dispatchKeyEvent(new KeyEvent(KeyEvent.ACTION_DOWN, 
         KeyEvent.KEYCODE_S)); 
         dispatchKeyEvent(new KeyEvent(KeyEvent.ACTION_UP, 
         KeyEvent.KEYCODE_S)); 
        } 
       }; 

       Runnable r2 = new Runnable() { 
        public void run() { 
         dispatchKeyEvent(new KeyEvent(KeyEvent.ACTION_DOWN, 
         KeyEvent.KEYCODE_DEL)); 
         dispatchKeyEvent(new KeyEvent(KeyEvent.ACTION_UP, 
         KeyEvent.KEYCODE_DEL)); 
        } 
       }; 
       while (true) { 

        runOnUiThread(r); 
        try { 
         Thread.sleep(200); 
        } catch (InterruptedException e) { 
         ExceptionHandler.handle(e); 
        } 
        runOnUiThread(r2); 
        try { 
         Thread.sleep(200); 
        } catch (InterruptedException e) { 
         ExceptionHandler.handle(e); 
        } 
       } 
      } 
     }).start(); 
    } 

這是我的適配器:

private class ArrayAdapterEinsatz extends DynamicArrayAdapter<Einsatz> { 

     public ArrayAdapterEinsatz(int textViewResourceId) { 
      super(ListEinsaetze.this, textViewResourceId, new ArrayList<Einsatz>()); 
     } 

     @Override 
     public View getView(final int position, View convertView, ViewGroup parent) { 

      Einsatz einsatz = getFilteredItems().get(position); 

      View rowView = convertView; 
      if (rowView == null) { 
       LayoutInflater inflater = ListEinsaetze.this.getLayoutInflater(); 
       rowView = inflater.inflate(R.layout.eintrag_einsatz, null); 
       ViewHolder viewHolder = new ViewHolder(); 

       viewHolder.tvId = (TextView) rowView.findViewById(R.id.eeinsatzid); 
       viewHolder.tvDatumVonbis = (TextView) rowView.findViewById(R.id.eeinsatzvonbis); 
       viewHolder.tvKundenName = (TextView) rowView.findViewById(R.id.eeinsatzkundenname); 
       viewHolder.tvKundenOrt = (TextView) rowView.findViewById(R.id.eeinsatzort); 
       viewHolder.tvEntfernung = (TextView) rowView.findViewById(R.id.eeinsatzentfernung); 
       viewHolder.iveRotate = (ImageViewEffects) rowView.findViewById(R.id.eeinsatzrotate); 
       viewHolder.iveRotate.setBitmap(Utils.drawableToBitmap(R.drawable.ic_refresh_1, 22), ImageViewEffects.ROTATE); 
       viewHolder.ivAnsprechpartnerVorhanden = (ImageView) rowView.findViewById(R.id.eeinsatzimageansprechvorhanden); 
       viewHolder.ivBelegeVorhanden = (ImageView) rowView.findViewById(R.id.eeinsatzimagebelegevorhanden); 
       viewHolder.ivTaetigkeitVorhanden = (ImageView) rowView.findViewById(R.id.eeinsatzimagetaetigkeitvorhanden); 

       rowView.setTag(viewHolder); 
      } 

      final ViewHolder holder = (ViewHolder) rowView.getTag(); 

      return rowView; 
     } 

    } 

最後,這是基本適配器類。注意最底層的實際過濾函數,它根據新的Random()。nextBoolean()進行過濾,這不能更簡單。

public class DynamicArrayAdapter<T> extends BaseAdapter implements Filterable { 

    public class DynamicAdapterFilter extends Filter { 

    private CharSequence m_Filter = null; 
    private List<T> m_FilteredItems = null; 

    public DynamicAdapterFilter() { 
     m_FilteredItems = new ArrayList<T>(); 
    } 

    @Override 
    protected FilterResults performFiltering(CharSequence constraint) { 


     FilterResults r = new FilterResults(); 
     List<T> items = null; 
     m_Filter = constraint; 

     if (constraint == null) { 
     items = m_AllItems; 
     } else { 
     items = m_FilteredItems; 
     items.clear(); 

     synchronized (SyncLock) { 
      List<T> l = new ArrayList<T>(m_AllItems); 
      for (T item : l) { 
      if (DynamicArrayAdapter.this.filter(item, constraint)) { 
       items.add(item); 
      } 
      } 
     } 
     } 

     r.values = items; 
     r.count = items.size(); 

     return r; 
    } 

    @SuppressWarnings("unchecked") 
    @Override 
    protected void publishResults(CharSequence constraint, FilterResults results) { 

     m_Items = (List<T>) results.values; 

     notifyDataSetChanged(); 

    } 

    public void refresh() { 
     if (m_Filter != null) { 

     filter(m_Filter); 
     } 
    } 

    public void refresh(FilterListener listener) { 
     if (m_Filter != null) { 

     filter(m_Filter, listener); 
     } 
    } 
    } 


    private final Object SyncLock = new Object(); 
    private Context m_Context = null; 
    private LayoutInflater m_Inflater = null; 
    private int m_DelegateResourceId; 
    private boolean m_ChangeNotificationsEnabled = true; 
    private List<T> m_AllItems = null; 
    private List<T> m_Items = null; 
    private DynamicAdapterFilter m_Filter = null; 


    public DynamicArrayAdapter(Context context, int delegateResourceId) { 
    this(context, delegateResourceId, new ArrayList<T>()); 
    } 

    public DynamicArrayAdapter(Context context, int delegateResourceId, List<T> items) { 
    m_Context = context; 
    m_DelegateResourceId = delegateResourceId; 
    initItems(items); 

    m_Inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE); 

    } 

    public void initItems(List<T> items) { 
    m_AllItems = m_Items = items; 
    } 

    /** 
    * returns the filtered item count (or all item count if no filter exists) 
    */ 
    @Override 
    public int getCount() { 
     return m_Items != null ? m_Items.size() : 0; 
    } 

    /** 
    * returns the item at the position in the filtered list (or all item count if 
    * no filter exists) 
    */ 
    @Override 
    public T getItem(int position) { 
    return m_Items.get(position); 
    } 

    /** 
    * holt position von item, exakte gleiche implementation wie ArrayAdapter<T> 
    * 
    * @param item 
    * @return 
    */ 
    public int getPosition(T item) { 
    return m_Items.indexOf(item); 
    } 

    /** 
    * {@inheritDoc} 
    */ 
    @Override 
    public long getItemId(int position) { 

    return position; 
    } 

    /** 
    * returns the list view item delegate view. this view is inflated if it isn't 
    * already inflated. Override this method to perform custom view operations 
    * for the list view item. 
    */ 
    @Override 
    public View getView(int position, View convertView, ViewGroup parent) { 
    if (convertView == null) { 
     convertView = m_Inflater.inflate(m_DelegateResourceId, parent, false); 
    } 

    return convertView; 
    } 

    /** 
    * returns the filter. this filter is created if it hasn't yet been created. 
    */ 
    @Override 
    public DynamicAdapterFilter getFilter() { 
    if (m_Filter == null) { 
     m_Filter = new DynamicAdapterFilter(); 
    } 

    return m_Filter; 
    } 

    /** 
    * If notifications are enabled, this notifies the base adapter that the 
    * backing data has changed 
    */ 
    @Override 
    public void notifyDataSetChanged() { 
    notifyDataSetChanged(false); 
    } 

    /** 
    * If forced or notifications are enabled, this notifies the base adapter that 
    * the backing data has changed 
    * 
    * @param force 
    *   force notify even if notifications are disabled 
    */ 
    public void notifyDataSetChanged(boolean force) { 
    if (force || m_ChangeNotificationsEnabled) { 

     super.notifyDataSetChanged(); 
    } 
    } 

    /** 
    * @return UI context 
    */ 
    public Context getContext() { 
    return m_Context; 
    } 

    /** 
    * @return UI inflater 
    */ 
    public LayoutInflater getLayoutInflater() { 
    return m_Inflater; 
    } 

    /** 
    * @return Layout ID of the list view item delegate 
    */ 
    public int getDelegateResourceId() { 
    return m_DelegateResourceId; 
    } 

    /** 
    * @return true if notifications are enabled, false otherwise 
    */ 
    public boolean getChangeNotificationsEnabled() { 
    return m_ChangeNotificationsEnabled; 
    } 

    /** 
    * Set whether notifications are enabled or not 
    */ 
    public void setChangeNotificationsEnabled(boolean value) { 
    m_ChangeNotificationsEnabled = value; 
    } 

    // internal mechanism to handle automagical updates of the list items 
    private void onDataChange() { 


    if (m_Filter == null) { 
     notifyDataSetChanged(); 
    } else { 
     m_Filter.refresh(); 
    } 
    } 

    /** 
    * @return All (unfiltered) items 
    */ 
    public List<T> getAllItems() { 
    return m_AllItems; 
    } 

    /** 
    * Set the backing list 
    */ 
    public void setAllItems(List<T> value) { 
    m_AllItems = value; 

    onDataChange(); 
    } 

    /** 
    * @return Current filtered items, or all items if there is no filter 
    */ 
    public List<T> getFilteredItems() { 
    return m_Items; 
    } 

    public void add(T item) { 
    synchronized (SyncLock) { 
     m_AllItems.add(item); 
    } 

    onDataChange(); 
    } 

    public void insert(T item, int index) { 
    synchronized (SyncLock) { 
     m_AllItems.add(index, item); 
    } 

    onDataChange(); 
    } 

    public void remove(T item) { 
    synchronized (SyncLock) { 
     m_AllItems.remove(item); 
    } 

    onDataChange(); 
    } 

    public void removeAt(int index) { 
    synchronized (SyncLock) { 
     m_AllItems.remove(index); 
    } 

    onDataChange(); 
    } 

    public void clear() { 
    synchronized (SyncLock) { 
     m_AllItems.clear(); 
    } 

    onDataChange(); 
    } 

    public void sort(Comparator<? super T> comparator) { 
    synchronized (SyncLock) { 
     Collections.sort(m_Items, comparator); 
    } 

    onDataChange(); 
    } 

    protected boolean filter(T item, CharSequence constraint) { // AR 
    return new Random().nextBoolean(); 

    } 
} 

所以基本上,程序每隔200毫秒觸發一次textwatcher。所以textwatcher一次又一次地過濾列表。節目中沒有別的東西在歡呼。

如果我沒有設置斷點,程序運行正常,沒有錯誤總是。

的錯誤,當我斷點的行

adapter.getFilter().filter(s.toString()); 

在TextWatcher並按住F8(繼續)按下發生時(這似乎把不規則性引入程序流程,我只能想象嗎?)。沒有斷點,不會發生錯誤。

而且順便說一句,如果我把

adapter.notifyDatasetChanged() 

adapter.getFilter().filter(s.toString()); 

同樣的錯誤發生的時間(我知道notifyDatasetChanged被稱爲publishResults 100%,所以我不 噸需要它,但它是否正確的行爲?)。

TL; DR:

是我的代碼(尤其是基礎適配器代碼)故障? (當然不能是別的不會發生錯誤)

編輯:

錯誤也沒有時,設置斷點我剛剛得到一個時...

+0

該代碼在多個線程運行,所以當你突破點即行我的猜測是,它會暫停這個線程,而其他人正在運行導致此錯誤。至於adapter.notifyDatasetChanged()方法beilg放在adapter.getFilter方法之後,我只能推測類似的事情正在發生。我不知道任何關於android開發的內容,所以我只是猜測這裏...... – Thihara 2013-03-27 10:23:08

+0

嗯,我堅信當調試器遇到斷點時,它會停止所有線程(=整個程序)。 – cdbeelala89 2013-03-27 10:27:12

+0

*如果我沒有設置斷點,程序運行良好,總是沒有錯誤。*和*錯誤也會在沒有設置斷點時發生我剛剛得到一個... *這兩句話似乎矛盾。你是否至少試圖在performFiltering方法中複製數據,就像我在前面的問題中推薦過你的那樣? – Luksprog 2013-03-27 10:40:41

回答

0

嘗試移動你的

adapter.getFilter().filter(s.toString()); 

在可運行的行中使用view.post()進入ui線程。

它「可能」解決您的問題。

+0

什麼是view.post()? – cdbeelala89 2013-03-27 12:41:29

+0

post是view.It的一個方法,它向ui線程發佈一個單獨的runnable。 – Ercan 2013-03-27 12:44:18

0

不要添加一個textwatcher而只是將適配器設置爲如下所示的編輯文本。

edSearch.setAdapter(adapter);

並覆蓋public Filter getFilter()和 protected void publishResults(CharSequence contraint,FilterResults results)
方法。

這將帶您problem..Faced的護理同樣的問題,它解決這樣:)