- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
为了理智起见,我将其发布在这里,因为我确信我现在无法自行解决。我发布了所有需要知道的东西,因为我认为为什么我过去没有解决这个问题是因为我没有发布所有内容。到目前为止,我已经剥离了代码,基本上它是尽可能轻的。但是仍然会发生此错误(尽管很少发生):
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 毫秒触发一次文本观察器。手动测试这个我会很疯狂:
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;
}
}
最后,这是基础适配器类。请注意最底部的实际过滤功能,它根据 new 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());
同样的错误将在 100% 的时间内发生(我知道在 publishResults 中调用了 notifyDatasetChanged,所以我不不需要它,但它仍然是正确的行为吗?)。
长话短说:
我的代码(尤其是基本适配器代码)是故障安全的吗? (否则不会发生任何错误)
编辑:
当没有设置断点时也会发生错误我刚得到一个...
最佳答案
试着移动你的
adapter.getFilter().filter(s.toString());
在 runnable 中使用 view.post() 进入 ui 线程。
它“可能”解决您的问题。
关于java - ListView 过滤器 : IllegalStateException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15656402/
我有一个坦克射击弹药的游戏。我对这部分代码的目标是检查它们是否与“碰撞”磁贴发生碰撞,如果是,则将其和磁贴移除。 代码如下所示,每 1/60 秒检查一次: Iterator iterator = sh
我尝试使用 JSR-303 注释(类级别)和验证器实现为 play 2.0.1 编写自定义表单验证器。 不幸的是,当我提交表单并且验证失败时,我收到了一个 IllegalStateException,
根据answer of BalusC ,我用过 FacesContext.getCurrentInstance().getExternalContext().redirect(url); 在我的 @P
这个问题已经有答案了: Copy a stream to avoid "stream has already been operated upon or closed" (10 个回答) 已关闭 5
这个问题已经有答案了: Spring: getOutputStream() has already been called for this response (3 个回答) 已关闭 4 年前。 我正
我正在尝试将 Activity 转换为 FragmentActivty 对象,以便获得 FragmentManager 对象 public class Main extends ListActivit
我正在尝试使用可编辑的组合框,通过用户的某些击键从数据库中快速搜索客户端的功能。我想要的是,用户将输入一些字母,如果这些字母与某些客户端匹配,这些客户端将保留在组合框的当前数据模型中。 代码如下。请修
这个问题已经有答案了: You need to use a Theme.AppCompat theme (or descendant) with this activity. Change to Th
我正在使用 Android Studio 和 Genymotion 作为模拟器创建一个应用程序,其中我在 3 个 EditText 中输入数据,当我单击按钮将其存储在 sqlite 数据库中时,它不起
我正在为 Android 构建一个简单的消息应用程序,并且在发送短信时遇到一些问题。我第一次使用 OnlickListener 时,消息被发送并显示在我的 ListView 中。当我在 Activit
我了解到 collect() 和 forEach() 都是流终端操作,在同一个流上调用它们会抛出 非法状态异常。但是,以下代码可以成功编译并按升序打印每个字符串的长度。不会引发任何异常。怎么会这样?
我对 classcastException 和非法状态异常都有点困惑,因为在大多数情况下它们看起来都很相似。 我在这个java代码中遇到了一个问题 class consumer {
我正在尝试这个小计算器程序。当我调用calculateResult()方法时,我想在第二个操作数为零且运算符为除法时显示IllegalStateException错误。但尽管如此,我在calculat
Stacktrace Here 导入java.util.*; 公共(public)类 AccountClient { public static void main(String[] args) {
我正在使用 readEntity() 方法读取 JAVAX 响应,但我收到以下堆栈跟踪: java.lang.IllegalStateException: Entity input stream ha
我是安卓新手。我正在尝试进行简单的登录 Activity ,但当我单击“登录”按钮时出现运行时错误。我认为我没有正确获取数据。我已经检查过,SQLite 中有一个与该 PK 相对应的数据。 日志猫。
我正在创建一个登录页面,工程师可以通过以“engg”开头的用户名登录。问题出在登录页面,当我使用正确的密码提供正确的输入时,它会给出“非法状态异常”。在错误的输入中,它工作正常。就像当我在我的 ora
我正在使用一些现有的 Java 设备驱动程序软件,该软件使用 JavaCOMM 进行串行 I/O。我昨天看到它抛出一个异常,其中有一个 IllegalStateException - 端口从 publ
这个问题不太可能对任何 future 的访客有帮助;它只与一个较小的地理区域、一个特定的时间点或一个非常狭窄的情况相关,通常不适用于全世界的互联网受众。如需帮助使此问题更广泛适用,visit the
我正在使用 Adventnet SNMPAPI 开发 UDP 监听程序。现在我需要将监听数据保存到数据库中。当我这样做时,我遇到了错误。任何人都可以帮忙解决这个问题吗... 这是我的代码。 impor
我是一名优秀的程序员,十分优秀!