[英]Java Publisher Sever chat program
我正在尝试创建一个具有一个发布者,一个服务器和多个订阅者的聊天应用程序。 发布者(发送到端口8000)将消息发送到服务器(在端口8000和5000上侦听),并将消息进一步转发给订户(在端口5000上侦听)。
到目前为止,我可以创建多个发布者,并且服务器和发布者之间的通信正常,但是,我无法将发布者发送的消息发送给订阅者
服务器端代码
package serverclient;
import java.io.*;
import java.net.*;
import java.util.logging.Level;
import java.util.logging.Logger;
public class Server extends Thread{
private Socket socket;
private int clientNumber;
public Server(Socket socket, int clientNumber){
this.socket = socket;
this.clientNumber = clientNumber;
if(socket.getLocalPort() == 5000)System.out.print("\nSubscriber "+ clientNumber +" is connected to the server");
if(socket.getLocalPort() == 8000)System.out.print("\nPublisher "+ clientNumber +" is connected to the server");
}
@Override
public void run(){
try {
BufferedReader dStream = new BufferedReader(new InputStreamReader(socket.getInputStream()));
PrintWriter out = new PrintWriter(socket.getOutputStream(), true);
while(true){
synchronized(this){
String clMessage = dStream.readLine();
System.out.println("\n"+clMessage);
// if(socket.getLocalPort() == 5000){
out.println("Hey the server is sending the message to subscriber");
// }
//out.println("Hey the publisher has sent the message : " + clMessage);
}
}
} catch (IOException ex) {
System.out.print("\nError has been handled 1\n");
}finally{
try {
socket.close();
} catch (IOException ex) {
System.out.print("\nError has been handled 2\n");
}
}
}
public static void main(String [] args) throws IOException{
int subNumber = 0;
int pubNumber = 0;
ServerSocket servSockpub = new ServerSocket(8000);
ServerSocket servSocksub = new ServerSocket(5000);
try {
while (true) {
Server servpub = new Server(servSockpub.accept(),++pubNumber);
servpub.start();
System.out.print("\nThe server is running on listen port "+ servSockpub.getLocalPort());
Server servsub = new Server(servSocksub.accept(),++subNumber);
servsub.start();
System.out.print("\nThe server is running on listen port "+ servSocksub.getLocalPort());
}
} finally {
servSockpub.close();
servSocksub.close();
}
}
}
发布者代码
package serverclient;
import java.net.*;
import java.io.*;
公共类发布者{公共静态void主(字符串[] args)抛出IOException {
Socket sock = new Socket("127.0.0.1",8000);
// reading from keyboard (keyRead object)
BufferedReader keyRead = new BufferedReader(new InputStreamReader(System.in));
// sending to client (pwrite object)
OutputStream ostream = sock.getOutputStream();
PrintWriter pwrite = new PrintWriter(ostream, true);
InputStream istream = sock.getInputStream();
BufferedReader receiveRead = new BufferedReader(new InputStreamReader(istream));
System.out.println("Start the chitchat, type and press Enter key");
String receiveMessage,sendMessage;
while(true)
{
sendMessage = keyRead.readLine(); // keyboard reading
pwrite.println(sendMessage); // sending to server
pwrite.flush(); // flush the data
if((receiveMessage = receiveRead.readLine()) != null) //receive from server
{
System.out.println(receiveMessage); // displaying at DOS prompt
}
else{
System.out.print("Null");
}
}
}
}
订户
package serverclient;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.net.Socket;
public class Subscriber {
public static void main (String [] args) throws IOException{
Socket sock = new Socket("127.0.0.1",5000);
// receiving from server ( receiveRead object)
InputStream istream = sock.getInputStream();
BufferedReader receiveRead = new BufferedReader(new InputStreamReader(istream));
System.out.println("Recive side");
String receiveMessage, sendMessage;
while(true)
{
System.out.print("Hey man " + receiveRead.readLine() + "\n");
if((receiveMessage = receiveRead.readLine()) != null) //receive from server
{
System.out.println(receiveMessage); // displaying at DOS prompt
}
else{
System.out.print("Null");
}
}
}
}
任何帮助表示赞赏。 我只想弄清楚为什么订户没有收到消息
有很多可能性可以处理实时通信问题。 我自己更喜欢使用事件/事件监听器。
当前,在您的程序中,服务器本身与处理订户连接的线程之间没有通信。
同样在一个侧节点上:即使发布者连接线程和订阅者连接线程之间进行了适当的通信,由于您使用的是同一个Server
类,所以现在将无法正常工作。 这不仅违反了单一职责原则,而且还会阻止服务器向订阅服务器发送消息。
假设您已建立连接,并且服务器类现在已与订阅服务器连接。 会发生什么?
订阅者将循环播放,直到其套接字的输入流上出现消息为止。 好,这正是我们想要的。 但是服务器做什么? 事实是完全一样的 。 Server的run
方法的try
块中的前几条语句是创建BufferedReader并从中读取直到收到消息。 现在,我们在每个站点上都有一个套接字,该套接字将无限期地等待某种消息的到达(显然,因为两者都在等待某件事,所以永远不会发生)。
为了防止这种情况,您应该首先检查流中是否有任何内容要读取:
while ( true )
{
if ( socket.getInputStream().available() != 0 )
{
// reading logic goes here....
synchronized ( this )
{
String clMessage = dStream.readLine();
System.out.println( "\n" + clMessage );
out.println( "Hey the server is sending the message to subscriber" );
}
}
// what shall be done when not reading.
}
现在第二部分。 如果要在线程之间进行通信,则需要实现一些逻辑。 如上所述,我喜欢侦听器的概念,因此我将展示一个使用侦听器的示例 :
MessageReceivedListener.java
import java.util.EventListener;
public interface MessageReceivedListener
extends EventListener
{
public void onMessageReceived( String message );
}
注意:接口不必扩展EventListener
因为EventListener
只是一个标记接口。 我本人仍然更愿意以此为提醒,说明界面的用途。
Server.java(节选)
// New constructor since we will pass a Listener now. Also new local variable for it.
public Server( Socket socket, int clientNumber, MessageReceivedListener mrl )
{
this.socket = socket;
this.clientNumber = clientNumber;
this.mrl = mrl;
if ( socket.getLocalPort() == 5000 )
System.out.print( "\nSubscriber " + clientNumber + " is connected to the server" );
if ( socket.getLocalPort() == 8000 )
System.out.print( "\nPublisher " + clientNumber + " is connected to the server" );
}
新的构造函数提供了一种将MessageReceivedListener传递到Server对象的方法。 或者,您也可以为其创建一个setter。
synchronized ( this )
{
String clMessage = dStream.readLine();
System.out.println( "\n" + clMessage );
out.println( "Hey the server is sending the message to subscriber" );
mrl.onMessageReceived( clMessage );
}
这就是魔术发生的地方。 收到消息后,我们将其传递给侦听器的onMessageReceived(String message)
方法。 但是它到底是做什么的呢? 这是我们在创建服务器对象时定义的。 这是两个示例,一个带有匿名类(Java 7和更低版本),另一个带有lambdas(Java 8和更高版本)。
示例Java 7和更早版本
Server servpub = new Server( servSockpub.accept(), ++pubNumber,
new MessageReceivedListener()
{
@Override
public void onMessageReceived( String message )
{
// call nother local method
// this method would need to be a static method of Server
// because it's in the scope of your server class
sendMessageToSubscribers(message);
}
} );
在这里,我们传递一个匿名类作为我们的MessageReceivedListener对象,并定义它的行为(在这种情况下,只需调用另一个将处理其余部分的方法即可。
现在,由于我们的MessageReceivedListener接口仅包含一种方法,我们也可以将其视为功能接口,因此使用lambda可以缩短代码并提高可读性。
Lambda示例(Java 8和更高版本)
Server servpub = new Server( servSockpub.accept(), ++pubNumber, Server::sendMessageToSubscribers);
在这种特定情况下,我们只有一个要传递给方法的参数,因此可以使用方法引用 。
如何实际实现方法sendMessageToSubs(String message)
由您决定。 但是,您需要跟踪已创建了多少个具有订阅者连接的线程,以及如何引用它们。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.