Не удается получить DataOutputStream сокетов в цикле для списка сокетов
Я пытаюсь создать клиент-серверную программу на Java, где клиенты отправляют простую строку на сервер, и каждый клиент, подключенный к серверу, получает эту строку. Поэтому я создал 2 серверных класса для сервера, которые создают новый поток для каждого клиента и слушают их. Также есть Список клиентских сокетов. Я хочу использовать его, чтобы отправить каждому клиенту строку, отправленную одним из клиентов. Для этого я использую цикл foreach. Но я получил Dataoutputstream только для клиента, который отправил строку на сервер:
public void run(){
try{
while(true){
String data = in.readUTF();
for(Socket soc : GlobalQdaServer.allClients){
DataOutputStream sOut = new DataOutputStream(soc.getOutputStream());
sOut.writeUTF(data + ". Total clients number in list: " + GlobalQdaServer.allClients.size());
}
}
}
catch(EOFException ex){
System.out.println(ex.getMessage());
}
catch(IOException ex){
System.out.println(ex.getMessage());
}
}
Таким образом, только этот клиент получает строку (которую он отправил сам). Другие ничего не получают. Это полные уроки:
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.EOFException;
import java.io.IOException;
import java.net.ServerSocket;
import java.net.*;
import java.util.ArrayList;
import java.util.List;
public class GlobalQdaServer {
public static List<Socket> allClients = new ArrayList<Socket>();
public static void main(String[] args) {
try{
int serverPort = 7896;
ServerSocket listenSocket = new ServerSocket(serverPort);
while(true){
Socket clientSocket = listenSocket.accept();
allClients.add(clientSocket);
Connection c = new Connection(clientSocket);
}
}
catch(IOException ex){
System.out.println("Server Socket creating failiure");
}
}
}
class Connection extends Thread{
DataInputStream in;
DataOutputStream out;
Socket clientSocket;
public Connection(Socket aClientSocket){
try{
clientSocket = aClientSocket;
in = new DataInputStream(clientSocket.getInputStream());
out = new DataOutputStream(clientSocket.getOutputStream());
this.start();
}
catch(IOException ex){
System.out.println(ex.getMessage());
}
}
public void run(){
try{
while(true){
String data = in.readUTF();
for(Socket soc : GlobalQdaServer.allClients){
DataOutputStream sOut = new DataOutputStream(soc.getOutputStream());
sOut.writeUTF(data + ". Total clients number in list: " + GlobalQdaServer.allClients.size());
}
}
}
catch(EOFException ex){
System.out.println(ex.getMessage());
}
catch(IOException ex){
System.out.println(ex.getMessage());
}
}
}
import java.io.BufferedReader;
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.EOFException;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.*;
public class GlobalQdaClient {
public static void main(String[] args) throws IOException {
int serverPort = 7896;
Socket s = new Socket("localhost", serverPort);
BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));
String input = "";
while(!input.equals("end")){
input = reader.readLine();
try{
DataInputStream in = new DataInputStream(s.getInputStream());
DataOutputStream out = new DataOutputStream(s.getOutputStream());
out.writeUTF(input);
String data = in.readUTF();
System.out.println("Recieved this: " + data);
}
catch(UnknownHostException ex){
System.out.println(ex.getMessage());
}
catch(EOFException ex){
System.out.println(ex.getMessage());
}
catch(IOException ex){
System.out.println(ex.getMessage());
}
}
}
}
Пожалуйста, помогите найти ход проблемы. Я новичок в розетках. Спасибо
1 ответ
Ваша программа на самом деле работает нормально.
Единственная проблема заключается в том, что вы блокируете клиентов непрерывным чтением.
Таким образом, вы фактически увидите сообщение, которое вы отправили другим клиентам после того, как они разблокируют чтение.
Попробуйте следующее:
- Запустить сервер
- Запустить Client1
- Запустить Client2
- Отправить сообщение от Client1 (обратите внимание, что Client2 теперь заблокирован из-за ввода)
- Отправить сообщение от Client2
Теперь вы увидите сообщение, отправленное ранее с client1 на client2.
Но если вам нужно, чтобы оба клиента получали сообщения после их отправки, вам нужно создать другой поток для непрерывной загрузки обновлений.
Я изменил ваш клиент, чтобы сделать это.
import java.io.BufferedReader;
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.EOFException;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.*;
import java.util.logging.Level;
import java.util.logging.Logger;
public class GlobalQdaClient extends Thread {
int serverPort = 7896;
BufferedReader reader ;
static String input = "";
Socket s;
DataInputStream in ;
DataOutputStream out ;
public GlobalQdaClient() {
try {
reader = new BufferedReader(new InputStreamReader(System.in));
s = new Socket("localhost", serverPort);
in = new DataInputStream(s.getInputStream());
out = new DataOutputStream(s.getOutputStream());
} catch (IOException ex) {
Logger.getLogger(GlobalQdaClient.class.getName()).log(Level.SEVERE, null, ex);
}
}
public static void main(String[] args) throws IOException {
GlobalQdaClient client = new GlobalQdaClient();
client.start();
while(!client.input.equals("end")){
client.input = client.reader.readLine();
try{
client.out.writeUTF(client.input);
if(client.in.available()>0){
String data = client.in.readUTF();
System.out.println("Recieved this: " + data);
}
}
catch(UnknownHostException ex){
System.out.println(ex.getMessage());
}
catch(EOFException ex){
System.out.println(ex.getMessage());
}
catch(IOException ex){
System.out.println(ex.getMessage());
}
}
}
@Override
public void run(){
try {
while(true){
String data = in.readUTF();
System.out.println("Recieved this: " + data);
}
} catch (IOException ex) {
Logger.getLogger(Message.class.getName()).log(Level.SEVERE, null, ex);
}
}
}