Параллельное программирование сокетов Java - PullRequest
0 голосов
/ 28 апреля 2018

Ниже приведен мой код для простого параллельного сервера. Всякий раз, когда я запускаю несколько клиентов, сервер печатает только ввод первого клиента. Я не уверен, что я сделал неправильно. Любая помощь будет оценена.

public static void main(String[] args) {
    try {
        ServerSocket serverSocket = new ServerSocket(8001);
        while (true){
            Socket clientSocket = serverSocket.accept();
            System.out.println(clientSocket);
            ConcurrentServer client = new ConcurrentServer(clientSocket);
            client.start();
        }
    } catch (IOException i){}
}

public void run(){
    try {
        inputStream = new BufferedReader(new InputStreamReader(concurrentSocket.getInputStream()));
        outputStream = new PrintWriter(new OutputStreamWriter(concurrentSocket.getOutputStream()));

        String testString = inputStream.readLine();
        System.out.println(testString);


    } catch (IOException i){}
}

Ответы [ 3 ]

0 голосов
/ 28 апреля 2018

Я действительно не знаю, зачем вам такая сложная структура входных и выходных потоков. Лучше использовать сканер, который будет ждать нового ввода. Также вы можете использовать PrintWriter для вывода результатов вашего разговора.

Вот сервер, который принимает несколько клиентов:

import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.Scanner;

public class ConcurrentServer extends Thread {
private Socket concurrentSocket;

public ConcurrentServer(Socket clientSocket) {
    this.concurrentSocket = clientSocket;
}

public static void main(String[] args) {
    try {
        ServerSocket serverSocket = new ServerSocket(8001);
        while (true){
            Socket clientSocket = serverSocket.accept();
            System.out.println(clientSocket);
            ConcurrentServer client = new ConcurrentServer(clientSocket);
            client.start();
        }
    } catch (IOException i){}
}

public void run(){
    try {
        InputStream inputStream = concurrentSocket.getInputStream();
        Scanner scanner = new Scanner(inputStream);
        OutputStream outputStream = concurrentSocket.getOutputStream();
        PrintWriter pw = new PrintWriter(outputStream);

        while(scanner.hasNextLine()){
            String line = scanner.nextLine();
            System.out.println(line);
            pw.println("message: " + line);
            pw.flush();
        }
    } catch (IOException i){}
}

}

0 голосов
/ 28 апреля 2018

Этот код может помочь вам понять, как запускать несколько клиентов одновременно. :)

Что делает этот код? TCP-клиент отправляет строку на сервер, а TCP-сервер отправляет обратно строку в формате UPPERCASE, и сервер может сделать это одновременно с несколькими подключениями.

У меня включены 3 файлы для сервера и еще один для тестирования сервера с несколькими клиентами (ClientTest.java)

Main.java

import java.io.IOException;

public class Main {

    public static void main(String[] args) {
        try {
            new Server(3000).start();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

Server.java

import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.logging.Logger;

public class Server {

    private ServerSocket sSocket;
    private boolean run;
    private int port;

    public Server(int port) throws IOException {
        this.port = port;
        this.sSocket = new ServerSocket(this.port);
    }

    public void start() {

        this.run = true;
        Logger.getLogger(getClass().getName()).info("Server is listening on port: " + port);

        try {
            while (run) {
                Socket cs = sSocket.accept();
                Logger.getLogger(getClass().getName())
                        .info("New Client Connected! " + cs.getPort());
                new Thread(new Client(cs)).start(); // Put to a new thread.
            }
        } catch (IOException e) {
            Logger.getLogger(getClass().getName()).severe(e.getMessage());
        }
    }

    public void stop() {
        this.run = false;
    }
}

Client.java (клиентский процесс на сервере)

import java.io.*;
import java.net.Socket;
import java.util.logging.Logger;

public class Client implements Runnable {

    private Socket clientSocket;

    private DataOutputStream out; // write for the client
    private BufferedReader in; // read from the client

    public Client(Socket clientSocket) {
        this.clientSocket = clientSocket;
    }

    @Override
    public void run() {
        // Do client process
        outToClient(inFromClient().toUpperCase());
        closeConnection();
    }

    private String inFromClient() {

        String messageFromClient = "";

        /*
         *  Do not use try with resources because once -
         *  - it exits the block it will close your client socket too.
         */
        try {
            in = new BufferedReader(new InputStreamReader(clientSocket.getInputStream()));
            messageFromClient = in.readLine();
        } catch (IOException e) {
            Logger.getLogger(getClass().getName()).severe("InFromClientErr - " + e.getMessage());
        }

        return messageFromClient.trim().equals("") ? "No Inputs given!" : messageFromClient;
    }

    private void outToClient(String message) {
        try {
            out = new DataOutputStream(clientSocket.getOutputStream());
            out.writeBytes(message);
        } catch (IOException e) {
            Logger.getLogger(getClass().getName()).severe("OutToClientErr - " + e.getMessage());
        }
    }

    private void closeConnection() {
        try {
            in.close();
            out.close();
            clientSocket.close();
        } catch (NullPointerException | IOException e) {
            Logger.getLogger(getClass().getName()).severe(e.getMessage());
        }
    }
}

ClientTest.java (для тестирования клиентов)

import java.io.*;
import java.net.Socket;
import java.util.Scanner;

public class ClientTest {

    public static void main(String[] args) {

        Socket clientSocket;

        try {
            clientSocket = new Socket("localhost", 3000);

            DataOutputStream outToServer = new DataOutputStream(clientSocket.getOutputStream());
            BufferedReader inFromServer = new BufferedReader(new InputStreamReader(clientSocket.getInputStream()));

            outToServer.writeBytes(new Scanner(System.in).nextLine() + '\n'); // Get user input and send.
            System.out.println(inFromServer.readLine()); // Print the server response.

        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}
0 голосов
/ 28 апреля 2018

Проблема была вместо этого с клиентом. Не сервер. Сокет был объявлен вне цикла for, и поэтому было создано только одно соединение. Как и ниже:

public static void main(String[] args) {
    try {
        socket = new Socket("127.0.0.1", 8001);

        for (int i = 0; i < 5; i++){
            System.out.println("Starting client: " + i);
            ConcurrentClient concurrentClient = new ConcurrentClient(socket, i);
            concurrentClient.run();
        }
    } catch (IOException io) {
    }
}

Сокет должен быть объявлен внутри цикла for следующим образом:

public static void main(String[] args) {
    try {
        for (int i = 0; i < 5; i++){
            socket = new Socket("127.0.0.1", 8001);
            System.out.println("Starting client: " + i);
            ConcurrentClient concurrentClient = new ConcurrentClient(socket, i);
            concurrentClient.run();
        }
    } catch (IOException io) {
    }
}
...