AndroidAsync - 从 WebSocket.StringCallback() 更新视图

AndroidAsync - updating views from WebSocket.StringCallback()

我正在编写一些小型聊天应用程序,并且我正在使用 AndroidAsync 在我的应用程序中获取 WebSocket 客户端功能。所以,问题是每当我尝试从 WebSocket.StringCallback().onStringAvailable(String) 修改我的 listView 时,它都会导致异常:

android.view.ViewRoot$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
            at android.view.ViewRoot.checkThread(ViewRoot.java:2932)
            at android.view.ViewRoot.focusableViewAvailable(ViewRoot.java:1712)
            at android.view.ViewGroup.focusableViewAvailable(ViewGroup.java:452)
            at android.view.ViewGroup.focusableViewAvailable(ViewGroup.java:452)
            at android.view.ViewGroup.focusableViewAvailable(ViewGroup.java:452)
            at android.view.ViewGroup.focusableViewAvailable(ViewGroup.java:452)
            at android.view.ViewGroup.focusableViewAvailable(ViewGroup.java:452)
            at android.view.View.setFlags(View.java:4614)
            at android.view.View.setFocusableInTouchMode(View.java:3190)
            at android.widget.AdapterView.checkFocus(AdapterView.java:694)
            at android.widget.AdapterView$AdapterDataSetObserver.onChanged(AdapterView.java:789)
            at android.database.DataSetObservable.notifyChanged(DataSetObservable.java:31)
            at android.widget.BaseAdapter.notifyDataSetChanged(BaseAdapter.java:50)
            at android.widget.ArrayAdapter.notifyDataSetChanged(ArrayAdapter.java:247)
            at persilabtest.zulfigarov.com.chatapp.ChatActivity.onStringAvailable(ChatActivity.java:76)
            at com.koushikdutta.async.http.WebSocketImpl.onMessage(WebSocketImpl.java:88)
            at com.koushikdutta.async.http.HybiParser.emitFrame(HybiParser.java:420)
            at com.koushikdutta.async.http.HybiParser.access0(HybiParser.java:46)
            at com.koushikdutta.async.http.HybiParser.onDataAvailable(HybiParser.java:197)
            at com.koushikdutta.async.DataEmitterReader.handlePendingData(DataEmitterReader.java:24)
            at com.koushikdutta.async.DataEmitterReader.onDataAvailable(DataEmitterReader.java:41)
            at com.koushikdutta.async.Util.emitAllData(Util.java:22)
            at com.koushikdutta.async.AsyncNetworkSocket.onReadable(AsyncNetworkSocket.java:146)
            at com.koushikdutta.async.AsyncServer.runLoop(AsyncServer.java:788)
            at com.koushikdutta.async.AsyncServer.run(AsyncServer.java:626)
            at com.koushikdutta.async.AsyncServer.access0(AsyncServer.java:41)
            at com.koushikdutta.async.AsyncServer.run(AsyncServer.java:568) 

我已经明白我无法从非 UI 线程修改视图。但我不知道在这种情况下我还能如何更新它们。这是我的 ChatActivity 代码:

public class ChatActivity extends ActionBarActivity
{
    private static final String WS_ADDRESS = "ws://192.168.0.106:8084/TestChatServer/chat";

    @InjectView(R.id.btnSend)
    Button btnSend;

    @InjectView(R.id.lvChat)
    ListView lvChat;

    @InjectView(R.id.etMsg)
    EditText etMsg;

    List<String> mMessages = new ArrayList<String>();

    ArrayAdapter<String> adapter;

    @Override
    protected void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_chat);
        ButterKnife.inject(this);
        adapter = new ArrayAdapter<String>(this, android.R.layout.simple_list_item_1, mMessages);
        lvChat.setAdapter(adapter);

        AsyncHttpClient.getDefaultInstance()
                .websocket(WS_ADDRESS, null, new AsyncHttpClient.WebSocketConnectCallback()
                {
                    @Override
                    public void onCompleted(Exception ex, WebSocket webSocket)
                    {

                        if (ex != null)
                        {
                            ex.printStackTrace();
                            return;
                        }
                        Log.d("myLogs", "OK");

                        webSocket.setStringCallback(new WebSocket.StringCallback()
                        {
                            @Override
                            public void onStringAvailable(String s)
                            {
                                String newStr = s.replaceAll("(#+\S+)", "<font color='#EE0000'></font>");
                                Log.d("myLogs", s + "\n" + newStr);
                                mMessages.add(newStr);
                                //This line causes the exception
                                adapter.notifyDataSetChanged();
                            }
                        });
                    }
                });

    }
} 

那么,有什么想法可以在每次发送新消息时更新我的​​ listView 吗?

您需要确保从 UI 线程更新视图。您可以像这样使用 Activity.runOnUiThread() 来做到这一点:

runOnUiThread(new Runnable() {
     @Override
     public void run() {

     //run your code that needs to update the UI here

    }
});

此错误的原因 Only the original thread that created a view hierarchy can touch its views 是因为它检测到另一个线程(除了创建视图的线程)正在尝试访问视图。如果多个资源试图同时更改某项内容,则会导致线程问题。

不要从没有主线程的运行 adapter.notifyDataSetChanged();,而是使用:

new Handler().post(new Runnable() {
    @Override
    public void run() {
        adapter.notifyDataSetChanged()
    }
});