ошибка: только исходный поток, создавший иерархию представлений, может касаться его представлений

Привет и спасибо, что посмотрели на мой вопрос. Я программист среднего уровня на C, но новичок в Android. Я пытался заставить работать программирование чата. Предполагая, что все остальное в приведенном ниже коде работает отлично. Мне нравится задавать один вопрос: когда я пытаюсь выполнить setText () из запущенного потока, я получаю исключение выше. Я просмотрел множество сайтов и здесь. Много чего нашел, но толком не понимаю. Пожалуйста, объясните мне как можно проще или предложите какое-нибудь простое решение, если это возможно.

Большое тебе спасибо!!

public class chatter extends Activity {

private String name = "Unknown User";

/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);



    final EditText msgToServer = (EditText) findViewById(R.id.msgBox);
    final EditText chatFromServer = (EditText) findViewById(R.id.chatBox); 

    final Button MsgToServer = (Button) findViewById(R.id.sendButton);

    Socket socket = null;
    String ipAddress = "192.168.1.103";
    try {
        InetAddress serverAddr = InetAddress.getByName(ipAddress);
        Socket socketMain = new Socket(serverAddr, 4444);
        socket = socketMain;
    } catch (IOException e) {
        // TODO Auto-generated catch block
        Log.e("TCP", "error", e);
    }

    final OutMsg outMsg = new OutMsg(socket);
    Thread msgSenderThread = new Thread(outMsg);
    msgSenderThread.start();

    //chatFromServer.post(new InMsg(socket, chatFromServer));
    Thread msgReceiverThread = new Thread(new InMsg(socket, chatFromServer));
    msgReceiverThread.start();

    MsgToServer.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            String msgToServerString; 
            msgToServerString = msgToServer.getText().toString();
            outMsg.message = name + ": " + msgToServerString;
            outMsg.readyToSend = true;
            msgToServer.setText("");
        }
    });
}

public void updateResultsInUi (String msg)
{
    final EditText chatFromServer = (EditText) findViewById(R.id.chatBox); 
    chatFromServer.setText(msg); 
}

public class InMsg implements Runnable {

    Socket socket;
    EditText chatFromServer;
    public InMsg(Socket socket, EditText chatFromServer)
    {
        this.socket = socket;
        this.chatFromServer = chatFromServer;
    }

    public void run(){
        try {
        BufferedReader in = new BufferedReader(new InputStreamReader(socket.getInputStream()));
        String str = "FIRSTMESSAGEFROMSERVER";
            while (true)
            {
                if (str.equals("FIRSTMESSAGEFROMSERVER"))
                    str = in.readLine();
                else
                    str = str + "\n" + in.readLine();
                Log.e("TCP", "got the message: " + str);
     //Here is where went wrong******************
                chatFromServer.setText(str);
     //******************************************
            }
        } catch (IOException e) {
            // TODO Auto-generated catch block
            Log.e("TCP", "error in receiving", e);
        }
    }

}

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    MenuInflater inflater = getMenuInflater();
    inflater.inflate(R.menu.menu, menu);
    return true;
}

@Override
public boolean onOptionsItemSelected(MenuItem item) {
    // Handle item selection
    switch (item.getItemId()) {
    case R.id.setNameMenu:
        setname();
        return true;
    default:
        return super.onOptionsItemSelected(item);
    }
}

public void populateChatBox (String msgFromS)
{
    Log.e("TCP", "going in to popC");
    final EditText textNameInput = (EditText) findViewById(R.id.nameBox);
    Log.e("TCP", " popC");
    textNameInput.setText(msgFromS);
    Log.e("TCP", "going out from popC");
}

public void setname()
{
    setContentView(R.layout.custom_dialog);
    final EditText textNameInput = (EditText) findViewById(R.id.nameBox);
    Button submitNameButton = (Button) findViewById(R.id.submitNameButton);
    submitNameButton.setOnClickListener(new OnClickListener() {
    @Override
        public void onClick(View v) {
        String nameinput = textNameInput.getText().toString();
            if (!name.equals(""))
                name = nameinput;
            setContentView(R.layout.main);
        }
    });
}
}

person Byte    schedule 23.02.2011    source источник


Ответы (5)


В вашем методе run ():

Message msg = new Message();
String textTochange = "text";
msg.obj = textTochange;
mHandler.sendMessage(msg);

Создайте mHandler в своем потоке пользовательского интерфейса;

Handler mHandler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            String text = (String)msg.obj;
            //call setText here
        }
};
person Veeresh    schedule 23.02.2011
comment
handleMessage, похоже, не является методом суперкласса для Handler; почему у вас @Override? Я получаю сообщение об ошибке, когда пытаюсь это сделать. - person shim; 24.03.2013

Вы не находитесь в потоке пользовательского интерфейса при установке текста. Вы должны быть в пользовательском интерфейсе, если хотите работать с элементами пользовательского интерфейса. Создайте обработчик сообщений в потоке пользовательского интерфейса, разместите в нем свои сообщения и вызовите setText из обработчика в потоке пользовательского интерфейса.

person Veeresh    schedule 23.02.2011
comment
да, я видел похожие ответы на другие вопросы, это слишком общий характер. Не то чтобы вы ошибались, но некоторые люди вроде меня немного медлительны. Но спасибо большое, я решил это - person Byte; 24.02.2011

вы можете сделать это всякий раз, когда находитесь в потоке:

msgToServer.post(new Runnable() {
    public void run() {
        msgToServer.setText("your text here");
    }
}
person zrgiu    schedule 23.02.2011
comment
Я думаю, вы меня неправильно поняли. Но я решил это с помощью обработчика. Спасибо за ваш ответ. - person Byte; 24.02.2011

Ваша проблема в том, что есть определенные взаимодействия, которые вы можете выполнять только в потоке пользовательского интерфейса. Это одна из них.

Похоже, вы можете захотеть использовать AsyncTask

http://developer.android.com/reference/android/os/AsyncTask.html

В принципе, вы могли бы вместо этого сделать свой Runnable AsyncTask и выполнить setText в onProgressUpdate, который запускается на UIThread.

person Mike dg    schedule 23.02.2011
comment
Я изучил это, но ничего на этой странице не имеет для меня смысла. Есть ли место, где я мог бы найти пример для AsyncTask? (Я решил это с помощью обработчика) Спасибо! - person Byte; 24.02.2011

Если вы хотите использовать AsyncTask для работы в фоновом режиме, я бы сделал это следующим образом:

public class UpdateTextProgress_Task extends AsyncTask<Void,String,Void> {
    Socket socket;
    EditText chatFromServer;    

    UpdateTextProgress_Task(EditText chatFromServer, Socket socket){
            this.socket = socket;
            this.chatFromServer = chatFromServer;
}

    @Override
    protected Void doInBackground(Void... params) {
       try {
               BufferedReader in = new BufferedReader(new InputStreamReader(socket.getInputStream()));
               String str = "FIRSTMESSAGEFROMSERVER";

               while(true){
                    if (str.equals("FIRSTMESSAGEFROMSERVER")){
                         str = in.readLine();
                    }
                    else{
                         str = str + "\n" + in.readLine();
                    }
                    Log.e("TCP", "got the message: " + str);

              publishProgress(str); // calls the onProgressUpdate method
          }catch (IOException e) {
        Log.e("UpdateTextProgress", e.getMessage());
    }
    return null;
}

@Override
protected void onProgressUpdate(String... progress) {
    chatFromServer.setText(progress[0]); //now we are on the UI thread so we can update our EditText

}

@Override
protected void onPostExecute(Void result) {
    super.onPostExecute(result);
    Log.e("UpdateTextProgress", "Finished");
}

}

Чтобы выполнить код:

UpdateTextProgress_Task task = new UpdateTextProgress_Task(chatFromServer,socket);
task.execute();
person Jordan Hochstetler    schedule 11.11.2013