View.setVisibility 中的 java.util.ConcurrentModificationException
Posted
技术标签:
【中文标题】View.setVisibility 中的 java.util.ConcurrentModificationException【英文标题】:java.util.ConcurrentModificationException in View.setVisibility 【发布时间】:2012-06-14 20:43:50 【问题描述】:我正在为视图实施拖放操作。开始拖动时,我将视图的可见性设置为INVISIBLE
,然后,如果拖动被中断 - 回到VISIBLE
:
public boolean onTouch(View v, MotionEvent event)
if (event.getAction() == MotionEvent.ACTION_DOWN)
// Skipped some code
boolean dragStarted = v.startDrag(data, shadowBuilder, v, 0);
if (dragStarted)
v.setVisibility(View.INVISIBLE)
还有:
if (event.getAction() == DragEvent.ACTION_DRAG_ENDED)
View droppedView = (View) event.getLocalState();
droppedView.setVisibility(View.VISIBLE);
当“拖动结束”事件被调用时,我得到了异常:
E/androidRuntime(7118): FATAL EXCEPTION: main
E/AndroidRuntime(7118): java.util.ConcurrentModificationException
E/AndroidRuntime(7118): at java.util.HashMap$HashIterator.nextEntry(HashMap.java:792)
E/AndroidRuntime(7118): at java.util.HashMap$KeyIterator.next(HashMap.java:819)
E/AndroidRuntime(7118): at android.view.ViewGroup.dispatchDragEvent(ViewGroup.java:1046)
E/AndroidRuntime(7118): at android.view.ViewGroup.dispatchDragEvent(ViewGroup.java:1048)
E/AndroidRuntime(7118): at android.view.ViewGroup.dispatchDragEvent(ViewGroup.java:1048)
E/AndroidRuntime(7118): at android.view.ViewGroup.dispatchDragEvent(ViewGroup.java:1048)
E/AndroidRuntime(7118): at android.view.ViewRootImpl.handleDragEvent(ViewRootImpl.java:3471)
E/AndroidRuntime(7118): at android.view.ViewRootImpl.handleMessage(ViewRootImpl.java:2620)
E/AndroidRuntime(7118): at android.os.Handler.dispatchMessage(Handler.java:99)
E/AndroidRuntime(7118): at android.os.Looper.loop(Looper.java:137)
E/AndroidRuntime(7118): at android.app.ActivityThread.main(ActivityThread.java:4575)
E/AndroidRuntime(7118): at java.lang.reflect.Method.invokeNative(Native Method)
E/AndroidRuntime(7118): at java.lang.reflect.Method.invoke(Method.java:511)
E/AndroidRuntime(7118): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:789)
E/AndroidRuntime(7118): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:556)
E/AndroidRuntime(7118): at dalvik.system.NativeStart.main(NativeMethod)
为什么以及如何解决它?
【问题讨论】:
只是一个想法,但是您尝试过 API 演示吗?他们在那里有一个很好的拖放示例。只需使用它并在您希望时准确更改可见性。因为我没有带有 API11+ 的设备,所以我无法检查我所做的任何事情是否有帮助(模拟器是不够的,因为它有时会给出错误的结果) 关于为什么,当您插入或删除可迭代的某些元素时,当您在迭代时会发生此异常。但它似乎是android中的一个错误。 试试droppedView.post(new Runnable() public void run() droppedView.setVisibility(View.VISIBLE););
你需要将 dropView 设为 final
我已经写了一些解释为什么会发生异常***.com/questions/9899382/…
【参考方案1】:
你可以试试这个
if (event.getAction() == DragEvent.ACTION_DRAG_ENDED)
final View droppedView = (View) event.getLocalState();
droppedView.post(new Runnable()
@Override
public void run()
droppedView.setVisibility(View.VISIBLE);
);
看起来 Android 本身试图在您结束拖动的同时访问 View 状态。
编辑
更准确的解释。通过设置setVisibility()
,您将在Android 内部视图集合中包含或排除View
,这些视图应该响应拖动事件。这个集合在拖动事件的调度过程中使用,因此通过尝试setVisibility
(换句话说,尝试修改拖动事件的侦听器)你会导致ConcurrentModificationException
【讨论】:
【参考方案2】:最好的办法是:
view.post(new Runnable()
public void run()
view.setVisibility(View.VISIBLE);
);
如果我们使用:
if (event.getAction() == DragEvent.ACTION_DRAG_ENDED)
final View droppedView = (View) event.getLocalState();
droppedView.post(new Runnable()
@Override
public void run()
droppedView.setVisibility(View.VISIBLE);
);
你仍然会强制关闭 Null.pointer。 ...
【讨论】:
【参考方案3】:也许this 可以提供帮助。这里在给定的链接中说:而不是DragEvent.ACTION_DRAG_ENDED
使用DragEvent.ACTION_DROP
。
【讨论】:
我需要在视图没有被丢弃的情况下处理取消拖放操作。以上是关于View.setVisibility 中的 java.util.ConcurrentModificationException的主要内容,如果未能解决你的问题,请参考以下文章