java.lang.IllegalStateException 对于 Android 版本 4.4.4
java.lang.IllegalStateException for Android version 4.4.4
我在 Android 中使用 adapters
,在极少数情况下,应用程序在为 ListView
设置数据时崩溃...请帮忙。
异常如下
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.
Make sure your adapter calls notifyDataSetChanged() when its content changes.
[in ListView(2131558745, class android.widget.ListView)
with Adapter(class com.adapters.CustomToBeSyncAdapter)
这是我的代码片段
private class ImportAsyncTask extends AsyncTask<Void, Integer, Integer> {
@Override
protected Integer doInBackground(Void... params) {
Realm realm = Realm.getDefaultInstance();
realm.executeTransaction(new Realm.Transaction() {
@Override
public void execute(Realm realm) {
try {
RealmResults<AssetsActivitiesRealm> mResultsActivities = realm.where(AssetsActivitiesRealm.class).findAll();
for (int i = 0; i < mResultsActivities.size(); i++) {
AssetsActivitiesRealm mAssetsActivitiePj = mResultsActivities.get(i);
SyncPojoList.add(new SyncPojo(mAssetsActivitiePj.getBatch(), mAssetsActivitiePj.getItemCode()));
}
} catch (Exception e) {
e.printStackTrace();
}
}
});
realm.close();
return 1;
}
@Override
protected void onPreExecute() {
showLoading();
}
@Override
protected void onPostExecute(Integer sum) {
try {
cancelLoading();
displayListView();
} catch (Exception e) {
e.printStackTrace();
}
}
}
private void notifyAdapter() {
getActivity().runOnUiThread(new Runnable() {
public void run() {
listViewSync.setAdapter(null);
if (mdataAdapter != null) {
mdataAdapter.notifyDataSetChanged();
listViewSync.requestLayout();
}
}
});
}
private void displayListView() {
mdataAdapter = new CustomToBeSyncAdapter(getActivity(),
R.layout.custom_tobesync_row, SyncPojoList);
notifyAdapter();
listViewSync.setAdapter(mdataAdapter);
listViewSync.invalidate();
}
当您在 AsyncTask#doInBackground()
中添加对象时,您仍在从后台线程更改 ListView
适配器的模型。
尝试在 for-loop
中使用 AsyncTask#publishProgress(Progress...)
,然后覆盖 AsyncTask#onProgressUpdate(Progress...)
,其中 Progress
是您的模型。
所以你的 for-loop
可能看起来像这样:
RealmResults<AssetsActivitiesRealm> mResultsActivities = realm.where(AssetsActivitiesRealm.class).findAll();
for (int i = 0; i < mResultsActivities.size(); i++) {
AssetsActivitiesRealm mAssetsActivitiePj = mResultsActivities.get(i);
publishProgress(new SyncPojo(mAssetsActivitiePj.getBatch(), mAssetsActivitiePj.getItemCode());
}
你的 onProgressUpdate
可能看起来像这样:
public void onProgressUpdate(SyncPojo progress) {
SyncPojoList.add(progress);
mdataAdapter.notifyDataSetChanged(); // Notifying changes to the adapter here.
}
不清楚为什么你需要在 AsyncTask
中 运行 所有这些,而 Realm 实际上宣称你并不需要在后台线程中 运行 查询,因为他们的查询速度如此之快(他们在获取数据时使用延迟加载),所以在大多数情况下它应该会影响 UI 线程——尤其是当您根本没有任何 "special" 查询时。
另一方面,您不需要创建 Realm.Transaction
来查询数据。事务仅用于确保更改数据时的正确性。
您绝对可以大大简化您的代码:
- 您在调用
notifyDatasetChanged
后在您的 ListView
上使用 requestLayout
,这可能没用。
- 为什么要将
try-catch
与 Exception
一起使用?您应该在代码导致异常之前对其进行处理,而不是仅仅在容易出错的代码周围添加 try-catch
。
- 您的
AsyncTask
returns 是一个无意义的整数,而不是实际返回结果。
- 设置适配器也会通知
ListView
进行更新,因此无需调用您的 notifyAdapter
。
notifyAdapter
方法做了很多它不需要做的事情 - 例如将适配器设置为 null
。
希望这对您有所帮助,否则我很乐意详细说明任何领域。
我在 Android 中使用 adapters
,在极少数情况下,应用程序在为 ListView
设置数据时崩溃...请帮忙。
异常如下
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.
Make sure your adapter calls notifyDataSetChanged() when its content changes.
[in ListView(2131558745, class android.widget.ListView)
with Adapter(class com.adapters.CustomToBeSyncAdapter)
这是我的代码片段
private class ImportAsyncTask extends AsyncTask<Void, Integer, Integer> {
@Override
protected Integer doInBackground(Void... params) {
Realm realm = Realm.getDefaultInstance();
realm.executeTransaction(new Realm.Transaction() {
@Override
public void execute(Realm realm) {
try {
RealmResults<AssetsActivitiesRealm> mResultsActivities = realm.where(AssetsActivitiesRealm.class).findAll();
for (int i = 0; i < mResultsActivities.size(); i++) {
AssetsActivitiesRealm mAssetsActivitiePj = mResultsActivities.get(i);
SyncPojoList.add(new SyncPojo(mAssetsActivitiePj.getBatch(), mAssetsActivitiePj.getItemCode()));
}
} catch (Exception e) {
e.printStackTrace();
}
}
});
realm.close();
return 1;
}
@Override
protected void onPreExecute() {
showLoading();
}
@Override
protected void onPostExecute(Integer sum) {
try {
cancelLoading();
displayListView();
} catch (Exception e) {
e.printStackTrace();
}
}
}
private void notifyAdapter() {
getActivity().runOnUiThread(new Runnable() {
public void run() {
listViewSync.setAdapter(null);
if (mdataAdapter != null) {
mdataAdapter.notifyDataSetChanged();
listViewSync.requestLayout();
}
}
});
}
private void displayListView() {
mdataAdapter = new CustomToBeSyncAdapter(getActivity(),
R.layout.custom_tobesync_row, SyncPojoList);
notifyAdapter();
listViewSync.setAdapter(mdataAdapter);
listViewSync.invalidate();
}
当您在 AsyncTask#doInBackground()
中添加对象时,您仍在从后台线程更改 ListView
适配器的模型。
尝试在 for-loop
中使用 AsyncTask#publishProgress(Progress...)
,然后覆盖 AsyncTask#onProgressUpdate(Progress...)
,其中 Progress
是您的模型。
所以你的 for-loop
可能看起来像这样:
RealmResults<AssetsActivitiesRealm> mResultsActivities = realm.where(AssetsActivitiesRealm.class).findAll();
for (int i = 0; i < mResultsActivities.size(); i++) {
AssetsActivitiesRealm mAssetsActivitiePj = mResultsActivities.get(i);
publishProgress(new SyncPojo(mAssetsActivitiePj.getBatch(), mAssetsActivitiePj.getItemCode());
}
你的 onProgressUpdate
可能看起来像这样:
public void onProgressUpdate(SyncPojo progress) {
SyncPojoList.add(progress);
mdataAdapter.notifyDataSetChanged(); // Notifying changes to the adapter here.
}
不清楚为什么你需要在 AsyncTask
中 运行 所有这些,而 Realm 实际上宣称你并不需要在后台线程中 运行 查询,因为他们的查询速度如此之快(他们在获取数据时使用延迟加载),所以在大多数情况下它应该会影响 UI 线程——尤其是当您根本没有任何 "special" 查询时。
另一方面,您不需要创建 Realm.Transaction
来查询数据。事务仅用于确保更改数据时的正确性。
您绝对可以大大简化您的代码:
- 您在调用
notifyDatasetChanged
后在您的ListView
上使用requestLayout
,这可能没用。 - 为什么要将
try-catch
与Exception
一起使用?您应该在代码导致异常之前对其进行处理,而不是仅仅在容易出错的代码周围添加try-catch
。 - 您的
AsyncTask
returns 是一个无意义的整数,而不是实际返回结果。 - 设置适配器也会通知
ListView
进行更新,因此无需调用您的notifyAdapter
。 notifyAdapter
方法做了很多它不需要做的事情 - 例如将适配器设置为null
。
希望这对您有所帮助,否则我很乐意详细说明任何领域。