我有一台运行Java JDK 7的Linux Microsoft Azure VM" Server" (使用SSH连接)
我的电脑(Mac OSX)正在运行Java JDK 8" Server"
消息客户端在我的计算机上编译并运行,服务器在我的虚拟机上运行和编译。我已将端口xxx2和xxx3配置为UDP。
服务器代码:
public class server {
public static void main(String args[]) throws Exception {
int OutPort = xxx2;
int InPort = xxx3;
byte data[] = new byte[2048];
System.out.println("LOCALHOST:" + InetAddress.getLocalHost());
InetAddress users[] = new InetAddress[1000];
DatagramSocket dataIn = new DatagramSocket(OutPort);
DatagramSocket dataOut = new DatagramSocket(InPort);
DatagramPacket dataIncoming = new DatagramPacket(data, 2048);
int userNum = 0;
while (true) {
try {
dataIn.receive(dataIncoming);
} catch (IOException e) {
System.out.println(e);
}
String message = dataIncoming.getData().toString();
if (message.startsWith("login")) {
users[userNum] = dataIncoming.getAddress();
System.out.println("NEW USER:" + dataIncoming.getAddress().toString() + ", ADDED AS:" + users[userNum]);
userNum++;
}
for (int i = 0; i <= userNum; i++) {
dataOut.send(new DatagramPacket(message.getBytes(), 2048, users[i], InPort));
}
}
}
}
客户代码:
public class message {
public String username;
public int OutPort = xxx2;
public int InPort = xxx3;
// xxx2 out, xxx3 in (default for testing the opposite).
final InetAddress address;
message() throws UnknownHostException {
// Create the variables.
InetAddress addresses[] = null;
address = InetAddress.getByName("xxxxx.cloudapp.net");
addresses = InetAddress.getAllByName("xxxxx.cloudapp.net");
// Create the colors.
Color background = new Color(141,234,184);
// Create the window.
JFrame window = new JFrame("Java Message");
window.setSize(600,400);
window.setLayout(new BorderLayout());
window.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
window.setVisible(true);
// Create the title.
JLabel title = new JLabel("Xxxxx's Java Messaging System");
title.setFont(new Font("Arial", Font.PLAIN, 24));
title.setOpaque(true);
title.setBackground(background);
title.setHorizontalAlignment(SwingConstants.CENTER);
window.add(title, BorderLayout.NORTH);
// Create the text output.
JTextArea content = new JTextArea();
content.setFont(new Font("Arial", Font.PLAIN, 14));
content.setEditable(false);
window.add(content, BorderLayout.CENTER);
JScrollPane scroll = new JScrollPane(content);
window.add(scroll, BorderLayout.CENTER);
// Create the info panel.
JTextArea info = new JTextArea();
info.setFont(new Font("Menlo", Font.PLAIN, 14));
startup(info, address, addresses);
window.add(info, BorderLayout.EAST);
// Create username dialog.
JDialog usernameWindow = new JDialog();
usernameWindow.setTitle("Username setup");
usernameWindow.setSize(300,200);
usernameWindow.setDefaultCloseOperation(JDialog.DO_NOTHING_ON_CLOSE);
usernameWindow.setVisible(true);
// Create username title.
JLabel loginTitle = new JLabel("Enter a username:");
loginTitle.setFont(new Font("Arial", Font.PLAIN, 16));
usernameWindow.add(loginTitle, BorderLayout.CENTER);
// Create username input.
JTextField usernameInput = new JTextField();
usernameInput.setFont(new Font("Arial", Font.PLAIN, 16));
usernameWindow.add(usernameInput, BorderLayout.SOUTH);
usernameInput.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent ae) {
if (usernameInput.getText().length() >= 8) {
errorDialog("Username too long, must be 8 characters or less!");
} else {
username = usernameInput.getText().toString().toUpperCase();
content.append(":Logged in as:" + username + "\n");
sendMessage("login", username, address);
usernameWindow.dispose();
}
}
});
// Create the text input.
JTextField messageInput = new JTextField();
window.add(messageInput, BorderLayout.SOUTH);
reciveMessages(content);
messageInput.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent ae) {
sendMessage(username, messageInput.getText(), address);
messageInput.setText("");
}
});
}
public static void main(String args[]) {
SwingUtilities.invokeLater(new Runnable() {
public void run() {
try {
new message();
} catch (Exception e) {
System.out.println("ERROR:" + e);
}
}
});
}
public void startup(JTextArea content, InetAddress ip, InetAddress ips[]) {
InetAddress local = null;
try {
local = InetAddress.getLocalHost();
} catch(Exception e) {
System.out.println(e);
}
content.append("SERVER IP(s):\n");
for (int i = 0; i < ips.length; i++) content.append(ips[i].getHostAddress() + "\n");
content.append("USING:\n" + ip.getHostAddress() + "\n\n");
content.append("LOCALHOST:\n" + local.getHostAddress() + "\n");
}
public void sendMessage(String username, String message, InetAddress address) {
byte data[] = new byte[2048];
char usernameChar[] = username.toCharArray();
char messageChar[] = message.toCharArray();
for (int i = 0; i < usernameChar.length; i++) data[i] = (byte)usernameChar[i];
data[usernameChar.length + 1] = (byte)':';
for (int c = 0; c < messageChar.length; c++) data[usernameChar.length + c + 2] = (byte)messageChar[c];
try {
DatagramSocket outSocket = new DatagramSocket(OutPort);
DatagramPacket dataOut = new DatagramPacket(data, 2048, address, InPort);
outSocket.send(dataOut);
} catch (Exception e) {
System.out.println(e);
}
}
public void reciveMessages(JTextArea content) {
new Thread("PortListener") {
public void run() {
try {
byte inData[] = new byte[2048];
DatagramSocket inSocket = new DatagramSocket(InPort);
DatagramPacket dataIn = new DatagramPacket(inData, 2048);
while (true) {
inSocket.receive(dataIn);
content.append(new String(dataIn.getData(), 0, dataIn.getLength()));
}
} catch (Exception e) {
System.out.println(e);
}
}
}.start();
}
public void errorDialog(String cause) {
}
}
问题:我在VM上运行服务器代码,控制台永远不会返回&#34; NEW USER:&#34;并且没有任何消息通过。
我对Java网络完全陌生,在完成Java指南一章(本书由Oracle出版)之后做了这个,我需要一些如何使其工作的指导。
我不知道这是一个简单的错误,还是整个想法和理论都是错误的。
我的理论:
服务器正在侦听数据包。
客户端和服务器数据[]大小不同。
服务器的IP地址不正确(例如:缺少内部IP),我需要使用SocketAddress。
我知道什么:
我已经编写了这个程序的一个版本,我编译了它,然后在切换端口的情况下重新编译它(因此传出成为传入数据)并在我的计算机上运行并且系统工作正常。
答案 0 :(得分:0)
好吧,所以在完成你的代码之后,我已经制作了一个正常工作的版本。这是您需要编辑的内容和原因(其中一些可能是不必要的,但这是使它适用于我的原因):
我可能已经对行号有所了解,因为我添加了一些打印和注释以帮助我理解代码。评论我是否遗漏了修复/没有明确解释。
服务器客户端:
在第9行,您不需要为数据包中指定的出站套接字指定端口。
DatagramSocket dataOut=new DatagramSocket();
在第18行,我不确定你希望通过调用数组的toString()方法完成什么,但你应该将字节数组转换回字符串,如下所示:
String message = new String(dataIncoming.getData());
在第24行,你将进入userNum,但userNum是要填充的数组的下一个位置的索引,因此没有任何内容。 将其更改为小于
for(int i=0;i<userNum;i++)
消息客户端:
第58行:toString()在字符串
上是多余的username=usernameInput.getText().toUpperCase();
第107行:就像服务器一样,你不需要声明端口,因为这是在数据包中完成的
DatagramSocket outSocket=new DatagramSocket();
第108行:由于您要将数据发送到服务器正在侦听的端口(出口),您应该发送到输出端,而不是输入端口
DatagramPacket dataOut=new DatagramPacket(data,2048,address,OutPort);
最后,第123行:我看到你想要做什么,但是仍然会给你一个长度为2048的字符串和许多不可读的字符。此外,该行不会被终止。使用trim()缩小到文本,使用\ n终止行。
content.append(new String(dataIn.getData()).trim()+"\n");