繁体   English   中英

Java .jar文件无法接收套接字消息

[英]Java .jar file can't receive socket message

我正在尝试创建服务器和客户端应用程序,服务器可以在其中向客户端发送消息,而客户端只能接受来自服务器的消息。

我已经实现了这个目标。 问题是,client.jar将不会从服务器收到任何消息。 但是在netbean中一切正常。

知道为什么会这样吗?

这是我完整的服务器代码:

public sServerUI() {
    super("Server : "+System.getenv("COMPUTERNAME")); // mendapatkan nama komputer
    initComponents();
}

public void startListener(){    
    Random randomGen = new Random();
    try{
        myPort = randomGen.nextInt(9999);
        server = new ServerSocket(myPort);//Bebas portnya, tp nggk boleh sudah terpakai atau pakai random jg bisa
        btnListen.setEnabled(false);
        while(key == null) {
            key = JOptionPane.showInputDialog(this, "Input your Key : ", "Insert Key", JOptionPane.PLAIN_MESSAGE);
        }
        if(key.equals("")) {
            key = "Random";
            txtMessage.setText(txtMessage.getText()+"Invalid key inputted, key automatically set to '"+key+"'\n");
        } else txtMessage.setText(txtMessage.getText()+"Key set to '"+key+"'\n");
    } catch (IOException e) {//Kalau sudah terpakai muncul error
        JOptionPane.showMessageDialog(this, "Could not listen at " + myPort);
        //Gagal, keluarin info
    } finally{
        myPort = server.getLocalPort();
        lblPort.setText("Port: "+myPort);
        System.out.println("Port: "+myPort);
    }

    acceptClient.start();
}
public void windowClosing(WindowEvent e){
    try {
        server.close();
        for (int i=0;i<numberOfClient;i++){
            socketIn[i].close();
            socketOut[i].close();
        }
    } catch (IOException ex) {
        System.out.println("Error "+ex.getMessage());
    }
}

class Accepter extends Thread{
    @Override
    public void run(){
        while (true){
            try{
                client[numberOfClient] = server.accept();
                numberOfClient++;
                lblStatus.setText("Status: "+numberOfClient+" client(s) connected");

                Handler handleClient = new Handler(numberOfClient-1);
                handleClient.start();
            } catch (IOException e) {
                JOptionPane.showMessageDialog(null, "Accept failed: " + myPort);
            }
        }
    }
}
class Handler extends Thread{
    private int arr;
    Handler(int ar){
        arr = ar;
        try{
            socketIn[arr] = new BufferedReader(new InputStreamReader(client[arr].getInputStream()));
            socketOut[arr] = new PrintWriter(client[arr].getOutputStream(), true);
        } catch (IOException e) {
            JOptionPane.showMessageDialog(null, "Read failed");
        }
    }

    @Override
    public void run(){
        while (true){
            try{
                if (socketIn[arr].ready()){
                    System.out.println("Reading...");
                    line = socketIn[arr].readLine();
                    if (!txtMessage.getText().equals("")){
                        txtMessage.setText(txtMessage.getText()+"\n");
                        //broadcast message ke client2 lain
                    }
                    txtMessage.setText(txtMessage.getText()+"Client "+(arr+1)+": "+line);

                    for (int i=0;i<numberOfClient;i++){
                        if (i!=arr){//jgn kembaliin ke client yg kirim
                            socketOut[i].println("Client "+(arr+1)+": "+line);
                        }
                    }
                }
            } catch (IOException e) {
                System.out.println("Read failed");
            }
        }
    }
}
private void btnListenActionPerformed(java.awt.event.ActionEvent evt) {                                          
    // TODO add your handling code here:
    startListener();
}
private void sendData(String data) {
    for (int j=0;j<numberOfClient;j++){
        socketOut[j].println(data);
    } 
}

private void btnSendActionPerformed(java.awt.event.ActionEvent evt) {                                        
    // TODO add your handling code here:
    if(numberOfClient > 0) {
        int packetIndex = 1;
        String ext = getExt(filePath, '.');
        String sData = "start"+pemisahString+fByte.length+pemisahString+ext+pemisahString; //menaruh kata kunci 'start' dan ukuran file di awal message, serta extensionnya
        sendData(sData);
        sData = "";
        int k = 0;
        for(int i = 0; i < fByte.length; i++) {
            if(k >= Math.ceil((double)fByte.length/10.0)) {
                k = 0;
                sData = rc4(key, sData);
                sendData(Integer.toString(packetIndex)+pemisahString+sData);
                txtMessage.setText(txtMessage.getText()+"packet-"+packetIndex+" sent ! isi : "+revertToString(rc4(key,revertToString(sData)))+"\n");
                packetIndex++;
                sData = "";
            }
            sData += fByte[i];
            sData += pemisahString;
            k++;
            if(i == fByte.length-1) {
                sData = rc4(key, sData);
                sendData(Integer.toString(packetIndex)+pemisahString+sData);
                txtMessage.setText(txtMessage.getText()+"packet-"+packetIndex+" sent ! isi : "+revertToString(rc4(key,revertToString(sData)))+"\n");
                packetIndex++;
                sData = "";
            }
        }
        sData = "end"+pemisahString;
        sendData(sData);
        txtMessage.setText(txtMessage.getText() + "Done ! divided into "+k+" piece(s) per packet\n");
    } else JOptionPane.showMessageDialog(this, "No Client Connected !", "Error", JOptionPane.ERROR_MESSAGE);
}                                       

private void fileBtnActionPerformed(java.awt.event.ActionEvent evt) {                                        
    // TODO add your handling code here:
    JFileChooser chooser = new JFileChooser();
    chooser.setCurrentDirectory(new File("d:/Kul/Smstr 6/Kripto n Steno/Stream Cipher/"));
    int returnVal = chooser.showOpenDialog(this);

    if(returnVal == JFileChooser.APPROVE_OPTION) {
        filePath = chooser.getSelectedFile().getPath();
        try {
            inputFile = new File(filePath);
            fIn = new FileInputStream(inputFile);
            fByte = new byte[(int)inputFile.length()];
            System.out.println("file size : "+(int)inputFile.length()+" byte(s)");
            System.out.print("Isi file : ");
            fIn.read(fByte);
            fIn.close();
            for(int i = 0; i < fByte.length; i ++) {
                System.out.print(fByte[i]+" ");
            }
            System.out.print("end of file\n");

            String stringBuatDitampilin = getExt(filePath, (char)92);

            txtMessage.setText(txtMessage.getText() + "'" + stringBuatDitampilin + "' Loaded !\n");
            btnSend.setEnabled(true);
            //fIn.close();
            //JOptionPane.showMessageDialog(this, "File Loaded !", "Success", JOptionPane.INFORMATION_MESSAGE);
        } catch(java.io.IOException e) {
            JOptionPane.showMessageDialog(this, e.toString(), "IO Error", JOptionPane.ERROR_MESSAGE);
        }
    }
}          
public static void main(String args[]) {
    /* Set the Nimbus look and feel */
    //<editor-fold defaultstate="collapsed" desc=" Look and feel setting code (optional) ">
    /* If Nimbus (introduced in Java SE 6) is not available, stay with the default look and feel.
     * For details see http://download.oracle.com/javase/tutorial/uiswing/lookandfeel/plaf.html 
     */
    try {
        for (javax.swing.UIManager.LookAndFeelInfo info : javax.swing.UIManager.getInstalledLookAndFeels()) {
            if ("Nimbus".equals(info.getName())) {
                javax.swing.UIManager.setLookAndFeel(info.getClassName());
                break;
            }
        }
    } catch (ClassNotFoundException | InstantiationException | IllegalAccessException | javax.swing.UnsupportedLookAndFeelException ex) {
        java.util.logging.Logger.getLogger(sServerUI.class.getName()).log(java.util.logging.Level.SEVERE, null, ex);
    }
    //</editor-fold>

    /* Create and display the form */
    java.awt.EventQueue.invokeLater(new Runnable() {
        @Override
        public void run() {
            new sServerUI().setVisible(true);
        }
    });
}

这是我完整的客户代码:

public sClientUI() {
    super("Client");
    initComponents();
}

public void listenSocket(){
//Create socket connection
    try{
        socket = new Socket(txtHost.getText(), Integer.parseInt(txtPort.getText()));
        socketOut = new PrintWriter(socket.getOutputStream(), true);
        socketIn = new BufferedReader(new InputStreamReader(socket.getInputStream()));
        txtMessage.setText(txtMessage.getText()+"Succesfully connected to "+txtHost.getText()+" !\n");
        while(key == null) {
            key = JOptionPane.showInputDialog(this, "Input your Key : ", "Insert Key", JOptionPane.PLAIN_MESSAGE);
        }
        if(key.equals("")) {
            key = "Random";
            txtMessage.setText(txtMessage.getText()+"Invalid key inputted, key automatically set to '"+key+"'\n");
        } else txtMessage.setText(txtMessage.getText()+"Key set to '"+key+"'\n");
        txtHost.setEditable(false);
        txtPort.setEditable(false);
        btnConnect.setEnabled(false);
        myListener = new Timer(250, readLine);
        myListener.start();
    } catch (UnknownHostException e) {
        JOptionPane.showMessageDialog(this, "Unknown host: "+e.getMessage(), "Unknown Hostname", JOptionPane.ERROR_MESSAGE);
    } catch  (IOException e) {
        JOptionPane.showMessageDialog(this, "Either your hostname is wrong, or you entered wrong port number\n"+e.getMessage(),"Input Error", JOptionPane.ERROR_MESSAGE);
    } catch (NumberFormatException e) {
        JOptionPane.showMessageDialog(this, e.getMessage(),"Invalid Port Number", JOptionPane.ERROR_MESSAGE);
    }
}

public void windowClosing(WindowEvent e){
    try {
        socket.close();
        socketIn.close();
        socketOut.close();
    } catch (IOException ex) {
        System.out.println("Error "+ex.getMessage());
    }
}
ActionListener readLine = new ActionListener(){
    @Override
    public void actionPerformed(ActionEvent ae) {
        try{
            if (socketIn.ready()){
                String tempReceiver;
                if((tempReceiver = socketIn.readLine()) != null) {
                    exStr(tempReceiver); //untuk memotong-motong string dan meng-create file baru
                }
            }
        } catch (IOException e) {
            System.out.println("Read failed");
        }
    }
};
private void btnConnectActionPerformed(java.awt.event.ActionEvent evt) {                                           
    // TODO add your handling code here:
    listenSocket();
}   
public static void main(String args[]) {
    /* Set the Nimbus look and feel */
    //<editor-fold defaultstate="collapsed" desc=" Look and feel setting code (optional) ">
    /* If Nimbus (introduced in Java SE 6) is not available, stay with the default look and feel.
     * For details see http://download.oracle.com/javase/tutorial/uiswing/lookandfeel/plaf.html 
     */
    try {
        for (javax.swing.UIManager.LookAndFeelInfo info : javax.swing.UIManager.getInstalledLookAndFeels()) {
            if ("Nimbus".equals(info.getName())) {
                javax.swing.UIManager.setLookAndFeel(info.getClassName());
                break;
            }
        }
    } catch (ClassNotFoundException | InstantiationException | IllegalAccessException | javax.swing.UnsupportedLookAndFeelException ex) {
        java.util.logging.Logger.getLogger(sClientUI.class.getName()).log(java.util.logging.Level.SEVERE, null, ex);
    }
    //</editor-fold>

    /* Create and display the form */
    java.awt.EventQueue.invokeLater(new Runnable() {
        @Override
        public void run() {
            new sClientUI().setVisible(true);
        }
    });
}

.jar客户端和服务器已相互连接,它们只是无法发送或接收任何消息

您的代码缺少一些重要的内容。 例如进口。 他们向我们展示了您使用了哪些库,并给了我们进行一些重新设计的机会。

1)例如一行:

myListener = new Timer(250, readLine);

不能为java.util.Timer,因为此类没有构造函数(long / int,ActionListener)。 可能此调用仅运行一次,而不是运行多次,但是我不知道并不能告诉您。

2)再举一个例子:

acceptClient.start();

我只能假设有一个实例变量

Accepter acceptClient = new Accepter();

但它也可以指代完全不同的类,它们的工作方式不同。

3)回到我的第一个例子:

您正在进行某种轮询。 这不是Java中的好样式,因为Java Thready经过设计优化,可以阻止线程。 请改用这样的代码块(未经测试;受服务器代码的启发)

class ClientThread extends Thread{
    @Override
    public void run(){
        try{
            String tempReceiver;
            while ((tempReceiver = socketIn.readLine()) != null) {
                exStr(tempReceiver);
            }
        } catch (IOException e) {
            System.out.println("Read failed");
        }
    }
}

在这种情况下,您不需要轮询,便会立即得到响应。 这会消耗更少的CPU。 完全不同的选择是使用Java NIO,它针对单线程应用程序进行了更优化(或者我们称之为少数线程)。

一些题外话:

OT 1)从线程中调用JOptionPane.showMessageDialog()setText()类的Swing函数很危险。 Swing本身不是线程安全的。 您应该在其周围使用EventQueue.invokeLater()包装器。 我最喜欢的: SwingUtils.invokeLater()

OT 2)具有变量tempReceiver class-global(我认为这是因为我看不到任何声明)也是危险的,因为您可能在其他地方使用了它,并且可能在调用中的readLine()exStr()期间将其覆盖了其他线程(如果Timer是Swing实现,它将运行一个额外的线程)。

OT 3)您说的是“ .jar客户端和服务器已相互连接”。 这是否意味着您有两个打开的控制台窗口,并且每个窗口中都运行着一个java -jar命令? 如果没有,您可能会错过一些重要的控制台输出(可能是例外)。 Windows的Java发行版有2个Java运行程序: javaw.exe ,当在资源管理器中双击.jar文件时调用。 java.exe也可以运行您的程序,但还会打开一个控制台窗口,该窗口显示System.out.println等的输出。该控制台对于调试非常重要。 它显示的输出与NetBeans或Eclipse中的“调试”窗口相同。

解决了

这是因为我使用某种密码算法对消息进行了加密,并且我将字节作为字符串发送给某些控制台无法读取的加密字符,因此程序无法执行某些关键操作来处理消息

还是谢谢你:)

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM