(小)Java服务器端应用程序无法接收DatagramPackets

时间:2015-01-13 00:07:03

标签: java azure udp port datagram

我有一台运行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出版)之后做了这个,我需要一些如何使其工作的指导。

我不知道这是一个简单的错误,还是整个想法和理论都是错误的。

我的理论:

  1. 服务器正在侦听数据包。

  2. 客户端和服务器数据[]大小不同。

  3. 服务器的IP地址不正确(例如:缺少内部IP),我需要使用SocketAddress。

  4. 我知道什么:

    我已经编写了这个程序的一个版本,我编译了它,然后在切换端口的情况下重新编译它(因此传出成为传入数据)并在我的计算机上运行并且系统工作正常。

1 个答案:

答案 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");