首頁  >  文章  >  Java  >  Java網路連通性如何測試

Java網路連通性如何測試

王林
王林轉載
2023-05-04 15:49:061230瀏覽

一般情況下,我們只需要判斷從一台機器是否可以存取(Ping)到另一台機器,此時,可以簡單的使用Java 類別庫中java.net.InetAddress 類別來實現,這個類別提供了兩個方法探測遠端機器是否可達

boolean isReachable(int timeout) // 测试地址是否可达 boolean isReachable(NetworkInterface netif, int ttl, int timeout)  // 测试地址是否可达.

簡單說來,上述方法就是透過遠端機器的IP 位址構造InetAddress 對象,然後呼叫其isReachable 方法,測試呼叫機器和遠端機器的網絡可達性。注意到遠端機器可能有多個 IP 位址,因而可能要迭代的測試所有的情況。

void isAddressAvailable(String ip){      try{        InetAddress address = InetAddress.getByName(ip);//ping this IP               if(address instanceof java.net.Inet4Address){           System.out.println(ip + " is ipv4 address");        }else         if(address instanceof java.net.Inet6Address){           System.out.println(ip + " is ipv6 address");        }else{           System.out.println(ip + " is unrecongized");        }               if(address.isReachable(5000)){            System.out.println("SUCCESS - ping " + IP + " with no interface specified");        }else{           System.out.println("FAILURE - ping " + IP + " with no interface specified");        }               System.out.println("\n-------Trying different interfaces--------\n");               Enumeration<NetworkInterface> netInterfaces =              NetworkInterface.getNetworkInterfaces();           while(netInterfaces.hasMoreElements()) {                NetworkInterface ni = netInterfaces.nextElement();                System.out.println( "Checking interface, DisplayName:" + ni.getDisplayName() + ", Name:" + ni.getName());      if(address.isReachable(ni, 0, 5000)){            System.out.println("SUCCESS - ping " + ip);        }else{            System.out.println("FAILURE - ping " + ip);        }               Enumeration<InetAddress> ips = ni.getInetAddresses();           while(ips.hasMoreElements()) {               System.out.println("IP: " + ips.nextElement().getHostAddress());          }        System.out.println("-------------------------------------------");        }          }catch(Exception e){        System.out.println("error occurs.");        e.printStackTrace();          }         }

程式輸出:

--------------START--------------    10.13.20.70 is ipv4 address   SUCCESS - ping 10.13.20.70 with no interface specified    -------Trying different interfaces--------    Checking interface, DisplayName:MS TCP Loopback interface, Name:lo   FAILURE - ping 10.13.20.70   IP: 127.0.0.1   -------------------------------------------   Checking interface, DisplayName:Intel(R) Centrino(R) Advanced-N 6200 AGN -   Teefer2 Miniport, Name:eth0   FAILURE - ping 10.13.20.70   IP: 9.123.231.40   -------------------------------------------   Checking interface, DisplayName:Intel(R) 82577LM Gigabit Network Connection -   Teefer2 Miniport, Name:eth2   SUCCESS - ping 10.13.20.70   -------------------------------------------   Checking interface, DisplayName:WAN (PPP/SLIP) Interface, Name:ppp0   SUCCESS - ping 10.13.20.70   IP: 10.0.50.189   -------------------------------------------    --------------END--------------

從上可以看出isReachable 的用法,可以不指定任何介面來判斷遠端網路的可達性,但這不能區分出封包是從那個網路介面發出去的(  如果本地有多個網路介面的話);而高階版本的isReachable  則可以指定從本地的哪個網路介面測試,這樣可以準確的知道遠端網路可以連通本地的哪個網路介面。

但是,Java 本身並沒有提供任何方法來判斷本地的哪個IP 位址可以連通遠端網絡,Java 網路程式介面也沒有提供方法來存取ICMP  協定封包,因而透過ICMP 的網路不可達數據包實現這一點也是不可能的( 當然可以用JNI 來實現,但就和系統平台相關了),  此時可以考慮本文下一節提出的方法。

在某些情況下,我們可能要確定本地的哪個網路位址可以連通遠端網絡,以便遠端網路可以回連到本地使用某些服務或發出某些通知。一個典型的應用場景 是,本地啟動了檔案傳輸服務( 如FTP),需要將本地的某個IP  位址傳送到遠端機器,以便遠端機器可以透過該位址下載檔案;或遠端機器提供某些服務,在某些事件發生時通知註冊了獲取這些事件的機器(  常見於系統管理領域),因而在註冊時需要提供本地的某個可達( 從遠端) 地址。

雖然我們可以用 InetAddress.isReachabl  方法判斷出本地的哪個網路介面可連通遠端玩過,但是由於單一網路介面是可以設定多個 IP 位址的,因而在此並不合適。我們可以使用 Socket  建立可能的 TCP 連接,進而判斷某個本地 IP 位址是否可達遠端網路。我們使用java.net.Socket 類別中的connect 方法

void connect(SocketAddress endpoint, int timeout)  //使用Socket连接服务器,指定超时的时间

這種方法需要遠端的某個端口,該端口可以是任何基於TCP 協定的開放服務的端口(如一般都會開放的ECHO 服務埠7, Linux 的 SSH 服務埠22 等)。實際上,建立的 TCP 連線被協定堆疊放置在連線佇列,進而分發到真正處理資料的各個應用服務,由於 UDP  沒有連線的過程,因而基於 UDP 的服務(如 SNMP)無法在此方法中應用。

具體過程是,列舉本地的每個網路位址,建立本地 Socket,在某個連接埠上嘗試連接遠端位址,如果可以連接上,則表示該本地位址可達遠端網路。

void printReachableIP(InetAddress remoteAddr, int port){      String retIP = null;           Enumeration<NetworkInterface> netInterfaces;      try{        netInterfaces = NetworkInterface.getNetworkInterfaces();        while(netInterfaces.hasMoreElements()) {               NetworkInterface ni = netInterfaces.nextElement();               Enumeration<InetAddress> localAddrs = ni.getInetAddresses();            while(localAddrs.hasMoreElements()){                InetAddress localAddr = localAddrs.nextElement();                if(isReachable(localAddr, remoteAddr, port, 5000)){                        retIP = localAddr.getHostAddress();                        break;               }        }          }      } catch(SocketException e) {          System.out.println(     "Error occurred while listing all the local network addresses.");      }         if(retIP == null){          System.out.println("NULL reachable local IP is found!");      }else{          System.out.println("Reachable local IP is found, it is " + retIP);      }          }         boolean isReachable(InetAddress localInetAddr, InetAddress remoteInetAddr,                     int port, int timeout) {       booleanisReachable = false;      Socket socket = null;      try{          socket = newSocket();          // 端口号设置为 0 表示在本地挑选一个可用端口进行连接         SocketAddress localSocketAddr = new InetSocketAddress(localInetAddr, 0);          socket.bind(localSocketAddr);          InetSocketAddress endpointSocketAddr =          new InetSocketAddress(remoteInetAddr, port);          socket.connect(endpointSocketAddr, timeout);                 System.out.println("SUCCESS - connection established! Local: " +            localInetAddr.getHostAddress() + " remote: " +            remoteInetAddr.getHostAddress() + " port" + port);          isReachable = true;      } catch(IOException e) {          System.out.println("FAILRE - CAN not connect! Local: " +        localInetAddr.getHostAddress() + " remote: " +        remoteInetAddr.getHostAddress() + " port" + port);      } finally{          if(socket != null) {          try{          socket.close();          } catch(IOException e) {             System.out.println("Error occurred while closing socket..");            }          }      }      return isReachable;   }

運行結果

--------------START--------------    FAILRE - CAN not connect! Local: 127.0.0.1 remote: 10.8.1.50 port22   FAILRE - CAN not connect! Local: 9.123.231.40 remote: 10.8.1.50 port22   SUCCESS - connection established! Local: 10.0.50.189 remote: 10.8.1.50 port22   Reachable local IP is found, it is 10.0.50.189   --------------END--------------

當網路環境中存在IPv4 和IPv6,也就是機器既有IPv4 位址,又有IPv6 位址的時候,我們可以對程式進行一些最佳化,例如

  • 由於IPv4 和IPv6 位址之間是無法互相存取的,因此只需要判斷IPv4 位址之間和IPv6 位址之間的可及性。

  • 對於IPv4 的換回位址可以不做判斷,對於IPv6 的Linklocal 位址也可以跳過測試

  • 根據實際的需要,我們可以優先選擇使用IPv4 或IPv6,提高判斷的效率

判斷本機位址和遠端位址是否同為IPv4 或IPv6

// 判断是 IPv4 还是 IPv6   if(!((localInetAddr instanceofInet4Address) && (remoteInetAddr instanceofInet4Address)   || (localInetAddr instanceofInet6Address) && (remoteInetAddr instanceofInet6Address))){   // 本地和远程不是同时是 IPv4 或者 IPv6,跳过这种情况,不作检测  break;   }

跳過本機位址和LinkLocal 地址

if( localAddr.isLoopbackAddress() ||       localAddr.isAnyLocalAddress() ||       localAddr.isLinkLocalAddress() ){       // 地址为本地环回地址,跳过      break;   }

以上是Java網路連通性如何測試的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:yisu.com。如有侵權,請聯絡admin@php.cn刪除