如何检查客户端是否已断开连接?

6

我有一个带有Socket的Java程序。我需要检查客户端是否已经断开连接。请给出一个如何操作的示例代码,以及对每个部分的说明。我已经进行了研究,但是仍然不明白。抱歉我的英语不太好。

以下是我的代码:

    package proov_server;

//SERVER 2
import java.io.*;
import java.net.*;

class server2 {
    InetAddress[] kasutaja_aadress = new InetAddress[1000];
    String newLine = System.getProperty("line.separator");
    int kliendiNr = 0;
    int kilene_kokku;
  server2(int port) {

    try {

      ServerSocket severi_pistik = new ServerSocket(port);
        System.out.println("Server töötab ja kuulab porti " + port + ".");


      while (true) {

        Socket pistik = severi_pistik.accept();
        kliendiNr++;

        kasutaja_aadress[kliendiNr] = pistik.getInetAddress();
        System.out.println(newLine+"Klient " + kliendiNr + " masinast "
                      + kasutaja_aadress[kliendiNr].getHostName() + " (IP:"
                      + kasutaja_aadress[kliendiNr].getHostAddress() + ")");

        // uue kliendi lõime loomine
        KliendiLoim klient = new KliendiLoim(pistik,kliendiNr);

        // kliendi lõime käivitamine
        klient.start();

      }
    }
    catch (Exception e) {
      System.out.println("Serveri erind: " + e);
    }
  }

  DataOutputStream[] väljund = new DataOutputStream[1000];
  DataInputStream[] sisend = new DataInputStream[1000];

  int klient = 0;
  int nr;
  // sisemine klass ühendusega tegelemiseks
      class KliendiLoim extends Thread {

        // kliendi pistik

      Socket pistik;
        // kliendi number


        KliendiLoim(Socket pistik2,int kliendiNr) {
          nr = kliendiNr;
          this.pistik = pistik2;

        }
        public boolean kontroll(){
            try{
                System.out.println("con "+pistik.isConnected());
                System.out.println("close "+pistik.isClosed());
                if(pistik.isConnected() &&  !pistik.isClosed()){
                    //System.out.print(con_klient);
                    return true;

                }
            }catch(NullPointerException a){
                System.out.println("Sihukest klienti pole!!!");

            }
            kliendiNr --;
            return false;

        }

        public void run() {
              try { 
                    sisend[nr] = new DataInputStream(pistik.getInputStream()); //sisend
                    väljund[nr] = new DataOutputStream(pistik.getOutputStream()); //väljund 
              }catch (Exception ea) {
                    System.out.println(" Tekkis erind: " + ea);
              }   
              while(true){
                      try{


                        BufferedReader br = new BufferedReader(new InputStreamReader(System.in));

                        System.out.print("Sisesta k2sk: ");
                        String k2sk = null;
                        k2sk = br.readLine();
                        /*
                        String command;
                        if(k2sk.indexOf(" ") < 0){
                            command = k2sk;
                        }else{
                            command = k2sk.substring(0, k2sk.indexOf(" "));
                        }
                        */
                        String[] words = k2sk.split("\\s+");
                        for (int i = 0; i < words.length; i++) {
                            words[i] = words[i].replaceAll(" ", "");
                        }
                        switch(words[0]){
                        case "suhtle":

                            if(väljund.length > klient && väljund[klient] != null)
                            {
                                väljund[klient].writeUTF("1");


                            }else{
                                väljund[klient] = null;
                                sisend[klient] = null;
                                System.out.println("Sihukest klienti pole");
                            }
                        break;
                        case "vaheta":
                            try{
                                int klinetnr = Integer.parseInt(words[1]);
                                //if(kontroll(klinetnr) ){
                                    klient = Integer.parseInt(words[1]);
                                //}
                            }
                            catch(NumberFormatException e){
                                System.out.println("See pole number!!! ");
                            }
                        break;

                        case "kliendid":
                            if(kliendiNr != 0){
                                for(int i=1;i <= kliendiNr;i++){
                                    if(kontroll()){
                                        System.out.println("Klient:"+i+" ip: " + kasutaja_aadress[i] );
                                    }else{
                                        System.out.println("Pisi");
                                        väljund[klient] = null;
                                        sisend[klient] = null;
                                    }
                                }
                                System.out.println(newLine);
                            }else{
                                System.out.println("Kiente pole");
                            }
                        break;

                        }
                      System.out.println(kliendiNr);
                    }catch(SocketException a){
                        System.out.println("Klient kadus");

                    }
                    catch(Exception e){
                         System.out.println(" Viga: " + e);
                    }

          }
        }



  }
  public static void main(String[] args) {
        new server2(4321);

  }
}

2
你能提供一些你的代码让我们帮忙吗? - jsalonen
1
我只是复制粘贴而已。我是Java的新手。 - Pisikoll
有很多代码,其中一些变量名称不是英文。我建议你翻译其中的一些变量名,这样阅读起来可能更容易。这不是必须的,但这可以帮助你。 - Marc-Andre
Socket文档中有一些看起来很有前途的方法,比如isClosed和isConnected。也许这可以作为一个开始? - Marc-Andre
@Marc-Andre 不,它们告诉您套接字的状态,而不是连接状态。 - user207421
@EJP 这只是一个建议,谢谢你指出来。 - Marc-Andre
2个回答

13

如果客户端已正确断开连接:

  • read()将返回-1
  • readLine()将返回null
  • 对于任何其他X,readXXX()将抛出EOFException

检测TCP连接丢失的唯一可靠方法是向其写入数据。最终,这将抛出IOException: connection reset,但由于缓冲区而需要至少两个写操作。


3
在Stackoverflow上有一个相关的帖子(点击这里),并提供了解决方案。基本上,该解决方案表示检测客户端-服务器断开连接的最佳方法是尝试从套接字读取数据。如果读取成功,则连接处于活动状态。如果在读取期间抛出异常,则表示客户端和服务器之间没有连接。或者,套接字可能使用超时值配置为完成读取操作。在这种情况下,如果超时时间已过,则会抛出套接字超时异常,可以将其视为客户端已断开连接或网络已关闭。
该帖子还谈到了使用isReachable方法 - 参见 InetAddress文档。但是,此方法仅告诉我们远程主机是否可达。这可能只是客户端从服务器断开连接的原因之一。您将无法使用此技术检测由于客户端崩溃或终止而导致的断开连接。

我在想如果套接字已连接但没有可读内容,会发生什么? - Marc-Andre
1
我对答案进行了微小的更正 - 如果套接字已连接并且没有任何内容可读,则不会抛出异常,程序将继续执行下一行。只有当套接字断开连接时,从套接字读取数据才会导致异常。 - Prahalad Deshpande
1
那不正确。如果在阻塞模式下设置了读取超时,并且没有数据,你将收到 SocketTimeoutException。由你决定这意味着对等方无活动或网络故障。你引用的主题中的答案质量非常低。 - user207421
@EJP感谢您的评论-我将您的注意事项纳入了我的答案中。上面发布的线程中被接受的答案对我来说似乎是合理的。但是,请更新此线程,以了解可用于解决此问题的任何更简单的技术。 - Prahalad Deshpande
失去可达性并不会导致对等方断开连接。它可能会中断连接,但这并不是同一件事。超时异常也不等同于对等方断开连接。检测客户端断开连接的最佳方法是读取并获取流的末尾,但检测中断连接的最佳方法是写入它。再次强调,这些不是同一件事。而问题是关于对等方断开连接,而不是关于中断连接。 - user207421

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接