Java Soket Threaded Server将消息发送回特定客户端

时间:2018-05-31 09:16:05

标签: java multithreading sockets server

我正在开发一个客户端 - 服务器Java GUI(Swing)应用程序,我已经成功地创建了从客户端接收消息的Threaded Server类,以及一个向服务器发送消息的Client类,Client和Server都是GUI应用程序,i我正在为我的互联网咖啡开发一个lan订购系统.. 我是套接字编程的新手,我现在需要的是当服务器GUI应用程序的用户通过单击消息到达时更改图标的JLabel查看GUI中的顺序时从服务器向客户端发送消息的方法。

那么如何从服务器向发送触发该特定JLabel的消息的特定客户端发送消息?以及如何向包含该消息的客户端用户显示JOptionpane(客户端有几个JPanel类)

发送消息的客户端类:

package questorderingsystem.engine;

import java.io.BufferedWriter; 
import java.io.IOException;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.net.InetAddress;
import java.net.Socket;
import java.net.UnknownHostException;
    public class SendItemAndIp {

    private static Socket socket;
    public static String message = "";
    public static void sendToServer(String item) throws UnknownHostException, IOException{
    InetAddress IP= InetAddress.getLocalHost();
    String ipadr = IP.toString();
    String PCNUM = ipadr.substring(ipadr.length() - 2);

    //IP SERVERAA 
    String host = "192.168.55.151";
    int port = 1978;
        InetAddress address = InetAddress.getByName(host);
        socket = new Socket(address, port);
        //Send the message to the server
        OutputStream os = socket.getOutputStream();
        OutputStreamWriter osw = new OutputStreamWriter(os);
        BufferedWriter bw = new BufferedWriter(osw);
        String oneLine = item.replace("\n", "%");
        String sendMessage = oneLine +"/"+ PCNUM;
        bw.write(sendMessage);
        bw.flush();
        socket.close();
}
}

线程服务器

package questorderingsystemserver;

import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;

public class ThreadedEchoServer {

static final int PORT = 1978;

public static void startServer() {
    ServerSocket serverSocket = null;
    Socket socket = null;

    try {
        serverSocket = new ServerSocket(PORT);
    } catch (IOException e) {
        e.printStackTrace();

    }
    while (true) {
        try {
            socket = serverSocket.accept();
        } catch (IOException e) {
            System.out.println("I/O error: " + e);
        }
        // new thread for a client
        new EchoThread(socket).start();
    }
}
}

EchoServer的

package questorderingsystemserver;

import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.DataOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.net.Socket;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.sound.sampled.LineUnavailableException;
import javax.sound.sampled.UnsupportedAudioFileException;
import javax.swing.JLabel;

public class EchoThread extends Thread {
protected Socket socket;



}
public EchoThread(Socket clientSocket) {
    this.socket = clientSocket;
}
public void run() {
    InputStream inp = null;
    BufferedReader brinp = null;
    DataOutputStream out = null;
    try {
        inp = socket.getInputStream();
        brinp = new BufferedReader(new InputStreamReader(inp));
        out = new DataOutputStream(socket.getOutputStream());
    } catch (IOException e) {
        return;
    }
    String line;
    while (true) {
        try {
            line = brinp.readLine();
            if ((line == null) || line.equalsIgnoreCase("QUIT")) {
                socket.close();
                return;
            } else {
                //do something

                out.flush();
            }
        } catch (IOException e) {
            e.printStackTrace();
            return;
        } catch (UnsupportedAudioFileException ex) {
            Logger.getLogger(EchoThread.class.getName()).log(Level.SEVERE, null, ex);
        } catch (LineUnavailableException ex) {
            Logger.getLogger(EchoThread.class.getName()).log(Level.SEVERE, null, ex);
        }
    }
}
}

2 个答案:

答案 0 :(得分:0)

套接字建立端到端连接,因此它连接到另一个特定的套接字。 ServerSocket略有不同。看看ServerSocket::accept的回归。返回的对象是Socket。不仅是一个套接字,而是直接连接到发出请求的客户端的套接字。由于您为每个EchoThread调用创建了一个新的ServerSocket#accept,因此每个EchoThread都有与其自己的客户端的连接。

您已经从套接字读取以检查连接是否已关闭或是否已发出“QUIT命令”。现在,您只需使用之前创建的else响应DataOutputStream - 子句:

while (true) {
    try {
        line = brinp.readLine();
        if ((line == null) || line.equalsIgnoreCase("QUIT")) {
            socket.close();
            return;
        } else {
            //writing here:
            out.write("Some response text\n".getBytes());
            out.flush();
        }
    } catch (IOException e) {
      ...
    } ...
}

但请注意,您的repsonse字符串需要以'\n'字符结尾,具体取决于您客户的“阅读逻辑”。

答案 1 :(得分:0)

你几乎完成了,因为你将第一个命令指定为“QUIT”,所以你可以按照自己的意愿实现更多的命令。

好的部分是使用BufferedReaderDataOutputStream来处理输入和输出。

请注意,如果您希望使用字符串进行客户端和服务器之间的通信,那么您可以转到PrintStreamPrintWriter而不是DataOutputStream

在线程服务器文件/类中,在try(现在安全)接受套接字之后放置此行new EchoThread(socket).start();

在客户端类/文件中,当您为服务器发送一些命令时,您可能也会期待一些结果。所以就像你创建的echo类一样,你可以为客户做同样的事情并处理响应。