gpt4 book ai didi

java - Android AsyncTask 中的 ConcurrentModificationException

转载 作者:行者123 更新时间:2023-12-01 21:40:47 30 4
gpt4 key购买 nike

我有一个奇怪的问题,我希望我能很好地解释。我的应用程序有两个 Activity - MainActivity 和 SearchActivity。我在 MainActivity 上有一个按钮,它会触发从设备上的数据库上传到我的网络服务器上的远程数据库。如果我在第一次启动应用程序时单击按钮,没问题,工作正常。如果我切换到 SearchActivity,不执行任何操作,然后切换回来,然后尝试该按钮,应用程序将崩溃并出现 ConcurrentModificationException。我有一个 AsyncTask,它发送本地数据库的内容(已经从数据库中取出并通过参数作为 ArrayList 发送到线程)。我花了几个小时调试它,但仍然无法弄清楚它在哪里。任何建议将不胜感激。

这是按下按钮时触发的代码,用于从单独的数据库线程请求数据库的内容

Button btnRemoteSync = (Button)findViewById(R.id.btnSync);
btnRemoteSync.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent startUpload = new Intent(getString(R.string.broadcast_search_database));
startUpload.putExtra("type-id",1);
LocalBroadcastManager.getInstance(getApplicationContext()).sendBroadcast(startUpload);
}
});

这是 BroadcastReceiver 中的代码,它从数据库管理器获取每个响应并将它们添加到自定义 ResponseObjects 的 ArrayList 中。当数据库线程发送 bssid 值 DONE 时,将启动 AsyncTask,并将 ArrayList 作为参数传入。

@Override
public void onReceive(Context context, Intent intent) {
String bssid = intent.getStringExtra(getString(R.string.data_bssid));
if (bssid.equals("DONE")) {
RemoteDatabaseUploader rdb = new RemoteDatabaseUploader(getApplicationContext());
rdb.execute(databases);
} else {
databases.add(new ResponseObject(getApplicationContext(),
bssid,
intent.getStringExtra(getString(R.string.data_ssid)),
intent.getStringExtra(getString(R.string.data_capabilities)),
intent.getIntExtra(getString(R.string.data_level), 0),
intent.getIntExtra(getString(R.string.data_frequency), 0),
intent.getStringExtra(getString(R.string.data_timestamp)),
intent.getDoubleExtra(getString(R.string.data_latitude), 0),
intent.getDoubleExtra(getString(R.string.data_longitude), 0)));

}
}

下面是 AsyncTask 的 doInBackground 代码

@Override
protected Integer doInBackground(ArrayList<ResponseObject>... params) {
ArrayList<ResponseObject> entries = params[0];
try {
URL url = new URL(insertURL);
for (Iterator<ResponseObject> it = entries.iterator(); it.hasNext();) {
ResponseObject ro = it.next(); // THIS IS WHERE THE EXCEPTION REFERENCES IN THE DEBUG OUTPUT
HttpURLConnection urlConnection = (HttpURLConnection)url.openConnection();

urlConnection.setRequestMethod("POST");
urlConnection.setRequestProperty("USER-AGENT", "Mozilla/5.0");
urlConnection.setRequestProperty("ACCEPT-LANGUAGE", "en-US,en;0.5");
urlConnection.setDoOutput(true);
String postParams = "bssid=" + ro.BSSID
+ "&ssid=" + ro.SSID
+ "&capabilities=" + ro.CAPABILITIES
+ "&level=" + String.valueOf(ro.LEVEL)
+ "&frequency=" + String.valueOf(ro.FREQUENCY)
+ "&timestamp=" + ro.TIMESTAMP
+ "&lat=" + String.valueOf(ro.LAT)
+ "&long=" + String.valueOf(ro.LON);
DataOutputStream wr = new DataOutputStream(urlConnection.getOutputStream());
wr.writeBytes(postParams);
wr.flush();
wr.close();
Log.d("RemoteDatabase : ", "Post sent " + ro.BSSID + " || " + String.valueOf(urlConnection.getResponseCode()));
}
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (ProtocolException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}

entries.clear();

return null;
}

<小时/>编辑 - 我似乎已将问题追溯到代码的另一部分,其中单击按钮时发送广播。该按钮肯定只发送一次(使用 Log.d 检查),但数据库中接收到的数据是接收两次。现在正在尝试解决此问题。

最佳答案

已经有一段时间了,但我意识到我忘记添加解决此问题的方法,以防对其他人有帮助。

我发现这里的问题实际上是某些 Android 设备发送广播的多个副本。我使用 HTC 手机进行测试,显然由于某种原因,他们发送了所有广播的 2 个副本。我的代码的工作方式是通过广播生成一个线程,这会导致 2 个相同的线程对相同的数据进行操作。当这些完成时,它们各自发送“完成”广播,这导致主线程收到其中 4 个广播。完全困惑。我最终必须为每个广播添加一个唯一的 ID token ,并在接收端记录这些值,因此,如果两次收到相同的 ID,则第二次不会采取任何操作。

关于java - Android AsyncTask 中的 ConcurrentModificationException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36499920/

30 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com