作者x049 (joker)
看板AndroidDev
標題[問題] java.lang.IllegalStateException
時間Tue Oct 6 23:44:19 2015
各位前輩大家好,小弟接觸android沒多久還是位新手,
目前在做作品時遇到了一個瓶頸,是在ListView.setAdapter()
短時間內大量重複更新時跳出以下錯誤:
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(2131296313, class android.widget.ListView) with
Adapter(class com.example.avalon.PlayerAdapter)]
以下是小弟寫的Adapter的程式碼:
public class PlayerAdapter extends BaseAdapter {
private LayoutInflater playersInflater;
public List<Player> playerslist;
public PlayerAdapter(Context context, List<Player> playerslist) {
// TODO 自動產生的建構子 Stub
if(playerslist!=null){
playersInflater=LayoutInflater.from(context);
this.playerslist=playerslist;
PlayerAdapter.this.notifyDataSetChanged();
}
}
@Override
public int getCount() {
// TODO 自動產生的方法 Stub
return (playerslist==null)? 0 : playerslist.size();
}
@Oerride
public Object getItem(int position) {
// TODO 自動產生的方法 Stub
return playerslist.get(position);
}
@Override
public long getItemId(int position) {
// TODO 自動產生的方法 Stub
return playerslist.indexOf(getItem(position));
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
// TODO 自動產生的方法 Stub
if(playerslist.size()!=0){
try{
convertView=playersInflater.inflate
(R.layout.players_list_item, null);
ImageView iv_Players_Status=(ImageView)
convertView.findViewById(R.id.IV_Players_Status);
TextView tv_Players_Order=(TextView)
convertView.findViewById(R.id.TV_Players_Order);
TextView tv_Players_Anonymous=(TextView)
convertView.findViewById(R.id.TV_Players_Anonymous);
iv_Players_Status.setImageResource
(playerslist.get(position).getStatus());
tv_Players_Order.setText(String.valueOf
(playerslist.get(position).getOrder()).toString());
tv_Players_Anonymous.setText
(playerslist.get(position).getAnonymity());
}catch(Exception e){
e.printStackTrace();
Log.e("text","PlayerAdapter="+e.toString());
}
return convertView;
}else{
convertView=playersInflater.inflate(R.layout.null_layout, null);
return convertView;
}
}
}
這幾天爬文有看到getCount()內寫成
return (playerslist==null)? 0 : playerslist.size();
還有在更新資料時呼叫notifyDataSetChanged()通知系統更新Adapter
可是在大量更新時app還是會崩潰跳出此錯誤
有聽人說可以在更新時隱藏按鈕,到更新完後在顯示按鈕
但是小弟是寫socket連伺服器,所以可以多人更新所以以上方法可能無法解決
還希望各位可以協助解決,謝謝~
--
※ 發信站: 批踢踢實業坊(ptt.cc), 來自: 1.175.134.228
※ 文章網址: https://www.ptt.cc/bbs/AndroidDev/M.1444146266.A.E50.html
推 chaos88218: Socket有沒有開另外一個Thread,有的話Socket跟UI資訊 10/07 01:20
→ chaos88218: 更新也很容易造成兩個Thread在協同資料上有困難,不 10/07 01:20
→ chaos88218: 知道是不是這個問題 10/07 01:20
→ chaos88218: 只要有thread影響到Ui執行緒就會跳出錯誤,handler在 10/07 01:28
→ chaos88218: 大量資訊處理也會因為資料順序更新問題使得ui執行緒 10/07 01:28
→ chaos88218: 收到指令時出問題 10/07 01:28
→ x049: 我有用一個Thread在背景做接收 10/07 12:37
→ x049: 接收完後再用runOnUiThread()做ListView.setAdapter() 10/07 12:38
→ x049: 之前有過出現null值的錯誤,所以我在接收時有用ArrayList<> 10/07 12:41
→ x049: 儲存背景接收到的資料再作setAdapter()才不被下一個接收覆蓋 10/07 12:43
→ x049: 之前因為這個問題在爬文時有看人說過 10/07 12:47
→ x049: 可能是在大量處理時adapter會改由Thread執行,而非UI 10/07 12:48
→ x049: 在呼叫notifyDataSetChanged()時才會產生該錯誤 10/07 12:48
→ x049: 只是不知道該如何解決 10/07 13:03
→ ssccg: 你背景thread和UI thread在動同一個adapter instance? 10/07 18:36
→ ssccg: 應該背景接到list後,UI thread把那個list內容copy到 10/07 18:37
→ ssccg: adapter裡的list,再notifyDataSetChanged就好吧 10/07 18:37
→ x049: 我背景只有做接收到ArrayList<> 10/07 22:04
→ x049: 而後來的存取ArrayList<>與adapter裡的list,再到 10/07 22:05
→ x049: 都是使用runOnUiThread()完成,而notifyDataSetChanged是放在 10/07 22:06
→ x049: adapter內,如文中程式這樣,我不確定是不是有用錯 10/07 22:06
→ x049: 目前只有在大量處理時會跳錯誤,我想說最多10個人同時點擊 10/07 22:10
→ x049: 所以為了測試,在伺服器寫了點擊一次送10次更改 10/07 22:11
→ x049: 在瘋狂點擊幾秒後才會出現這個錯誤 10/07 22:11
→ corrupt003: getItemId() 可直接回傳position 10/07 23:01
→ corrupt003: 你怎麼把更新過的資料copy到adapter的? 10/07 23:02
→ corrupt003: 要記得arraylist裡的東西都是物件,直接用 = 是指到同 10/07 23:04
→ corrupt003: 一個reference 10/07 23:04
→ corrupt003: this.playerslist=playerslist; 你建構子中的這行就 10/07 23:06
→ corrupt003: 很可能出問題 10/07 23:06
→ corrupt003: 你也可以改用或參考ArrayAdapter的程式碼,它是 threa 10/07 23:16
→ corrupt003: d-safe 10/07 23:16
→ x049: 感謝大大,改成new一個List<Player>再用addAll()複製過去 10/07 23:49
→ x049: 解決這個錯誤了,只是我在remove時貌似出現新的錯誤 10/07 23:51
→ x049: java.util.ConcurrentModificationException 10/07 23:51
→ corrupt003: 有用Iterator嗎?這個錯誤google一下有不少人討論 10/07 23:54
→ x049: 抱歉剛剛以解決,原來是我用Log勘查ArrayList<> 10/07 23:57
→ x049: 每次存取都查看ArrayList<>累積多少,可能是有些被Thread刪除 10/07 23:58
→ x049: 才會跳出這個錯誤 10/07 23:58