#java #android #сокет
Я хотел бы спросить про сокеты в Android. Я перерыл половину интернета и не нашел
того, что мне нужно.
Задача написать клиент под Android на Java, который бы отправлял и получал сообщения.
Когда писал на компьютер, все было чудесно, тут же столкнулся с проблемой многопоточности
(нельзя вызывать сетевые задачи из главного потока). Я более менее разобрался, заставил
отправлять сообщения на сервер, но никак не получается сделать, чтобы TextView сам
принимал и обновлял историю диалога.
Вот код клиента, сервер даже переписал как простой эхо сервер, отсылает обратно то,
что отправил клиент. Ниже код клиента, помогите с реализацией приема сообщений от сервера.
import android.os.AsyncTask;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.net.Socket;
public class MainActivity extends AppCompatActivity {
private Socket socket;
private Button SendBut;
private EditText editText;
private TextView textView;
private PrintWriter os;
private BufferedReader is;
networking net;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
SendBut = (Button) findViewById(R.id.SendBut);
editText = (EditText) findViewById(R.id.editText);
textView = (TextView) findViewById(R.id.TxtFrSrv);
net = new networking();
net.execute();
}
public void onClck(View view) {
String txt = null;
txt = editText.getText().toString();
os.println(txt);
}
public class networking extends AsyncTask < String, Void, Void > {
@Override
protected Void doInBackground(String...params) {
try {
socket = new Socket("10.0.2.2", 4444);
os = new PrintWriter(new PrintWriter(new OutputStreamWriter(socket.getOutputStream())),
true);
} catch (IOException e) {
e.printStackTrace();
}
return null;
}
}
}
Пытался изобразить что то похожее на это, но с нулевым значение запустить не получается...
public class getMsg extends Thread{
public void run(){
try {
is=new BufferedReader(new InputStreamReader(socket.getInputStream()));
final String finc;
finc=is.readLine();
while (!finc.equals("break")){
MainActivity.this.runOnUiThread(new Runnable() {
@Override
public void run() {
textView.setText(finc);
}
});
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
Ответы
Ответ 1
Use retrofit Luke! Retrofit+rxJava+retrolambda Допустим ответ от сервера такой {"text":"текст сообщения"} Как обрабатываем public class Message{ @SerializedName("text") public String message; } public interface RequestApi { String BASE_URL = "http://server/"; RequestApi api = new Retrofit.Builder() .baseUrl(RequestApi.BASE_URL) .addConverterFactory(GsonConverterFactory.create()) .addCallAdapterFactory(RxJavaCallAdapterFactory.create()) .subscribeOn(AndroidSchedulers.mainThread() .build() .create(RequestApi.class); @GET("?method=getMsg") ObservablegetMessage(); } public class MainActivity extends Activity{ @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); SendBut = (Button) findViewById(R.id.SendBut); editText = (EditText) findViewById(R.id.editText); textView = (TextView) findViewById(R.id.TxtFrSrv); RequestApi.api.getMessage().subscribe(m->textView.setText(m.message)); } } Ответ 2
Вот решение вашей задачи private class Networking extends AsyncTask{ @Override protected String doInBackground(String... params) { try { Socket socket = new Socket("10.0.2.2", 4444); PrintWriter writer = new PrintWriter(new PrintWriter(new OutputStreamWriter(socket.getOutputStream())), true); writer.write(params[0]); writer.flush(); BufferedReader reader = new BufferedReader(new InputStreamReader(socket.getInputStream())); StringBuilder result = new StringBuilder(); String line; while ( (line = reader.readLine()) != null) { result.append(line); } return result.toString(); } catch (IOException e) { e.printStackTrace(); } return null; } @Override protected void onPostExecute(String s) { textView.setText(s); } } И вызывать AsyncTask необходимо из метода onClick() new Networking().execute(text); Ответ 3
public void GetMsg(){ new Thread(new Runnable() { @Override public void run() { try { socket=new Socket(Host,4444); } catch (IOException e) { e.printStackTrace(); } try{ is = new BufferedReader(new InputStreamReader(socket.getInputStream())); } catch (IOException e) { e.printStackTrace(); } while (true){ String msg = null; try { msg=is.readLine(); } catch (IOException e) { e.printStackTrace(); } if (msg==null){ break; } else { displayMsg(msg); } } } }).start(); } public void displayMsg(String msg){ final String msg1 = msg; handler.post(new Runnable() { @Override public void run() { appendTextAndScroll(msg1); } }); } Всем большое спасибо за помощь, я немного переделал клиент и решил данную проблему вот таким кодом, надеюсь кому-нибудь это будет полезно. Метод GetMsg вызываю в onCreate().Ответ 4
А почему ты хочешь, чтобы это делал TextView? Как ты себе это представляешь? TextView - это всего лишь объект лейаута. Если ты создаешь какой-то writer, os = new PrintWriter(new PrintWriter(new OutputStreamWriter(socket.getOutputStream())), true); то очевидно ты хочешь чтобы он тебе писал в исходящий поток- getOutputStream(). Если тебе нужно получать информацию, то скорее всего тебе нужен reader, который тебе ее читать будет из входящего потока - getInputStream().Ответ 5
Я бы сделал цикл с ожиданием Thread.sleep(400) и проверял, пришло ли сообщение, если сообщение !=null, то publishProgress(message) и в onProgressUpdate (! Точно не помню как называется метод, его нужно переопределить в AsyncTask) в ссылку на TextView, которую нужно заранее инициализировать, забивал текст. Если я правильно понял суть задачи
Комментариев нет:
Отправить комментарий