ошибка: только исходный поток, создавший иерархию представлений, может касаться его представлений - PullRequest
14 голосов
/ 24 февраля 2011

Привет и спасибо, что посмотрели на мой вопрос.Я программист среднего уровня в 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);
        }
    });
}
}

Ответы [ 5 ]

31 голосов
/ 24 февраля 2011

В вашем методе 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
        }
};
3 голосов
/ 24 февраля 2011

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

1 голос
/ 24 февраля 2011

вы можете сделать это, когда вы в потоке:

msgToServer.post(new Runnable() {
    public void run() {
        msgToServer.setText("your text here");
    }
}
0 голосов
/ 11 ноября 2013

Если вы хотите использовать 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();
0 голосов
/ 24 февраля 2011

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

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

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

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

...