日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關(guān)咨詢(xún)
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問(wèn)題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營(yíng)銷(xiāo)解決方案
使用Java測(cè)試網(wǎng)絡(luò)連通性的幾種方法

在網(wǎng)絡(luò)編程中,有時(shí)我們需要判斷兩臺(tái)機(jī)器之間的連通性,或者說(shuō)是一臺(tái)機(jī)器到另一臺(tái)機(jī)器的網(wǎng)絡(luò)可達(dá)性。在系統(tǒng)層面的測(cè)試中,我們常常用 Ping 命令來(lái)做驗(yàn)證。盡管 Java 提供了比較豐富的網(wǎng)絡(luò)編程類(lèi)庫(kù)(包括在應(yīng)用層的基于 URL 的網(wǎng)絡(luò)資源讀取,基于 TCP/IP 層的 Socket 編程,以及一些輔助的類(lèi)庫(kù)),但是沒(méi)有直接提供類(lèi)似 Ping 命令來(lái)測(cè)試網(wǎng)絡(luò)連通性的方法。本文將介紹如何通過(guò) Java 已有的 API,編程實(shí)現(xiàn)各種場(chǎng)景下兩臺(tái)機(jī)器之間的網(wǎng)絡(luò)可達(dá)性判斷。在下面的章節(jié)中,我們會(huì)使用 Java 網(wǎng)絡(luò)編程的一些類(lèi)庫(kù) java.net.InetAddress 和 java.net.Socket,通過(guò)例子解釋如何模擬 Ping 命令。

創(chuàng)新互聯(lián)公司2013年至今,先為楊浦等服務(wù)建站,楊浦等地企業(yè),進(jìn)行企業(yè)商務(wù)咨詢(xún)服務(wù)。為楊浦企業(yè)網(wǎng)站制作PC+手機(jī)+微官網(wǎng)三網(wǎng)同步一站式服務(wù)解決您的所有建站問(wèn)題。

一般情況下,我們僅僅需要判斷從一臺(tái)機(jī)器是否可以訪問(wèn)(Ping)到另一臺(tái)機(jī)器,此時(shí),可以簡(jiǎn)單的使用 Java 類(lèi)庫(kù)中 java.net.InetAddress 類(lèi)來(lái)實(shí)現(xiàn),這個(gè)類(lèi)提供了兩個(gè)方法探測(cè)遠(yuǎn)程機(jī)器是否可達(dá)

 
 
  1. boolean isReachable(int timeout) // 測(cè)試地址是否可達(dá) 
  2. boolean isReachable(NetworkInterface netif, int ttl, int timeout)  
  3. // 測(cè)試地址是否可達(dá). 

簡(jiǎn)單說(shuō)來(lái),上述方法就是通過(guò)遠(yuǎn)端機(jī)器的 IP 地址構(gòu)造 InetAddress 對(duì)象,然后調(diào)用其 isReachable 方法,測(cè)試調(diào)用機(jī)器和遠(yuǎn)端機(jī)器的網(wǎng)絡(luò)可達(dá)性。注意到遠(yuǎn)端機(jī)器可能有多個(gè) IP 地址,因而可能要迭代的測(cè)試所有的情況。

 
 
  1. void isAddressAvailable(String ip){  
  2.     try{  
  3.       InetAddress address = InetAddress.getByName(ip);//ping this IP  
  4.        
  5.       if(address instanceof java.net.Inet4Address){  
  6.          System.out.println(ip + " is ipv4 address");  
  7.       }else 
  8.         if(address instanceof java.net.Inet6Address){  
  9.          System.out.println(ip + " is ipv6 address");  
  10.       }else{  
  11.          System.out.println(ip + " is unrecongized");  
  12.       }  
  13.        
  14.       if(address.isReachable(5000)){  
  15.           System.out.println("SUCCESS - ping " + IP + " with no interface specified");  
  16.       }else{  
  17.          System.out.println("FAILURE - ping " + IP + " with no interface specified");  
  18.       }  
  19.        
  20.       System.out.println("\n-------Trying different interfaces--------\n");  
  21.        
  22.       Enumeration netInterfaces =  
  23.             NetworkInterface.getNetworkInterfaces();     
  24.       while(netInterfaces.hasMoreElements()) {     
  25.            NetworkInterface ni = netInterfaces.nextElement();     
  26.            System.out.println( 
  27. "Checking interface, DisplayName:" + ni.getDisplayName() + ", Name:" + ni.getName()); 
  28.      if(address.isReachable(ni, 0, 5000)){  
  29.           System.out.println("SUCCESS - ping " + ip);  
  30.       }else{  
  31.           System.out.println("FAILURE - ping " + ip);  
  32.       }  
  33.        
  34.       Enumeration ips = ni.getInetAddresses();     
  35.       while(ips.hasMoreElements()) {     
  36.           System.out.println("IP: " + ips.nextElement().getHostAddress());    
  37.       }  
  38.       System.out.println("-------------------------------------------");  
  39.       }  
  40.         }catch(Exception e){  
  41.       System.out.println("error occurs.");  
  42.       e.printStackTrace();  
  43.         }        
  44.  } 

程序輸出:

 
 
  1. --------------START--------------  
  2.  
  3.  10.13.20.70 is ipv4 address  
  4.  SUCCESS - ping 10.13.20.70 with no interface specified  
  5.  
  6.  -------Trying different interfaces--------  
  7.  
  8.  Checking interface, DisplayName:MS TCP Loopback interface, Name:lo  
  9.  FAILURE - ping 10.13.20.70  
  10.  IP: 127.0.0.1  
  11.  -------------------------------------------  
  12.  Checking interface, DisplayName:Intel(R) Centrino(R) Advanced-N 6200 AGN -  
  13.  Teefer2 Miniport, Name:eth0  
  14.  FAILURE - ping 10.13.20.70  
  15.  IP: 9.123.231.40  
  16.  -------------------------------------------  
  17.  Checking interface, DisplayName:Intel(R) 82577LM Gigabit Network Connection -  
  18.  Teefer2 Miniport, Name:eth1  
  19.  SUCCESS - ping 10.13.20.70  
  20.  -------------------------------------------  
  21.  Checking interface, DisplayName:WAN (PPP/SLIP) Interface, Name:ppp0  
  22.  SUCCESS - ping 10.13.20.70  
  23.  IP: 10.0.50.189  
  24.  -------------------------------------------  
  25.  
  26.  --------------END-------------- 

從上可以看出 isReachable 的用法,可以不指定任何接口來(lái)判斷遠(yuǎn)端網(wǎng)絡(luò)的可達(dá)性,但這不能區(qū)分出數(shù)據(jù)包是從那個(gè)網(wǎng)絡(luò)接口發(fā)出去的 ( 如果本地有多個(gè)網(wǎng)絡(luò)接口的話 );而高級(jí)版本的 isReachable 則可以指定從本地的哪個(gè)網(wǎng)絡(luò)接口測(cè)試,這樣可以準(zhǔn)確的知道遠(yuǎn)端網(wǎng)絡(luò)可以連通本地的哪個(gè)網(wǎng)絡(luò)接口。

但是,Java 本身沒(méi)有提供任何方法來(lái)判斷本地的哪個(gè) IP 地址可以連通遠(yuǎn)端網(wǎng)絡(luò),Java 網(wǎng)絡(luò)編程接口也沒(méi)有提供方法來(lái)訪問(wèn) ICMP 協(xié)議數(shù)據(jù)包,因而通過(guò) ICMP 的網(wǎng)絡(luò)不可達(dá)數(shù)據(jù)包實(shí)現(xiàn)這一點(diǎn)也是不可能的 ( 當(dāng)然可以用 JNI 來(lái)實(shí)現(xiàn),但就和系統(tǒng)平臺(tái)相關(guān)了 ), 此時(shí)可以考慮本文下一節(jié)提出的方法。

在某些情況下,我們可能要確定本地的哪個(gè)網(wǎng)絡(luò)地址可以連通遠(yuǎn)程網(wǎng)絡(luò),以便遠(yuǎn)程網(wǎng)絡(luò)可以回連到本地使用某些服務(wù)或發(fā)出某些通知。一個(gè)典型的應(yīng)用場(chǎng)景 是,本地啟動(dòng)了文件傳輸服務(wù) ( 如 FTP),需要將本地的某個(gè) IP 地址發(fā)送到遠(yuǎn)端機(jī)器,以便遠(yuǎn)端機(jī)器可以通過(guò)該地址下載文件;或者遠(yuǎn)端機(jī)器提供某些服務(wù),在某些事件發(fā)生時(shí)通知注冊(cè)了獲取這些事件的機(jī)器 ( 常見(jiàn)于系統(tǒng)管理領(lǐng)域 ),因而在注冊(cè)時(shí)需要提供本地的某個(gè)可達(dá) ( 從遠(yuǎn)端 ) 地址。

雖然我們可以用 InetAddress.isReachabl 方法判斷出本地的哪個(gè)網(wǎng)絡(luò)接口可連通遠(yuǎn)程玩過(guò),但是由于單個(gè)網(wǎng)絡(luò)接口是可以配置多個(gè) IP 地址的,因而在此并不合適。我們可以使用 Socket 建立可能的 TCP 連接,進(jìn)而判斷某個(gè)本地 IP 地址是否可達(dá)遠(yuǎn)程網(wǎng)絡(luò)。我們使用 java.net.Socket 類(lèi)中的 connect 方法

 
 
  1. void connect(SocketAddress endpoint, int timeout)  //使用Socket連接服務(wù)器,指定超時(shí)的時(shí)間 

這種方法需要遠(yuǎn)程的某個(gè)端口,該端口可以是任何基于 TCP 協(xié)議的開(kāi)放服務(wù)的端口(如一般都會(huì)開(kāi)放的 ECHO 服務(wù)端口 7, Linux 的 SSH 服務(wù)端口 22 等)。實(shí)際上,建立的 TCP 連接被協(xié)議棧放置在連接隊(duì)列,進(jìn)而分發(fā)到真正處理數(shù)據(jù)的各個(gè)應(yīng)用服務(wù),由于 UDP 沒(méi)有連接的過(guò)程,因而基于 UDP 的服務(wù)(如 SNMP)無(wú)法在此方法中應(yīng)用。

具體過(guò)程是,枚舉本地的每個(gè)網(wǎng)絡(luò)地址,建立本地 Socket,在某個(gè)端口上嘗試連接遠(yuǎn)程地址,如果可以連接上,則說(shuō)明該本地地址可達(dá)遠(yuǎn)程網(wǎng)絡(luò)。

 
 
  1. void printReachableIP(InetAddress remoteAddr, int port){  
  2.     String retIP = null;  
  3.      
  4.     Enumeration netInterfaces;  
  5.     try{  
  6.       netInterfaces = NetworkInterface.getNetworkInterfaces();  
  7.       while(netInterfaces.hasMoreElements()) {     
  8.           NetworkInterface ni = netInterfaces.nextElement();     
  9.           Enumeration localAddrs = ni.getInetAddresses();  
  10.           while(localAddrs.hasMoreElements()){  
  11.               InetAddress localAddr = localAddrs.nextElement();  
  12.               if(isReachable(localAddr, remoteAddr, port, 5000)){  
  13.                       retIP = localAddr.getHostAddress();  
  14.                       break;         
  15.       }  
  16.       }  
  17.         }  
  18.     } catch(SocketException e) {  
  19.         System.out.println( 
  20.     "Error occurred while listing all the local network addresses.");  
  21.     }     
  22.     if(retIP == null){  
  23.         System.out.println("NULL reachable local IP is found!");  
  24.     }else{  
  25.         System.out.println("Reachable local IP is found, it is " + retIP);  
  26.     }         
  27.  }   
  28.      
  29.  boolean isReachable(InetAddress localInetAddr, InetAddress remoteInetAddr,  
  30.                    int port, int timeout) {  
  31.  
  32.     booleanisReachable = false;  
  33.     Socket socket = null;  
  34.     try{  
  35.         socket = newSocket();  
  36.         // 端口號(hào)設(shè)置為 0 表示在本地挑選一個(gè)可用端口進(jìn)行連接 
  37.         SocketAddress localSocketAddr = new InetSocketAddress(localInetAddr, 0);  
  38.         socket.bind(localSocketAddr);  
  39.         InetSocketAddress endpointSocketAddr =  
  40.         new InetSocketAddress(remoteInetAddr, port);  
  41.         socket.connect(endpointSocketAddr, timeout);         
  42.         System.out.println("SUCCESS - connection established! Local: " +  
  43.           localInetAddr.getHostAddress() + " remote: " +  
  44.           remoteInetAddr.getHostAddress() + " port" + port);  
  45.         isReachable = true;  
  46.     } catch(IOException e) {  
  47.         System.out.println("FAILRE - CAN not connect! Local: " +  
  48.       localInetAddr.getHostAddress() + " remote: " +  
  49.       remoteInetAddr.getHostAddress() + " port" + port);  
  50.     } finally{  
  51.         if(socket != null) {  
  52.         try{  
  53.         socket.close();  
  54.         } catch(IOException e) {  
  55.            System.out.println("Error occurred while closing socket..");  
  56.           }  
  57.         }  
  58.     }  
  59.     return isReachable;  
  60.  } 

運(yùn)行結(jié)果

 
 
  1. --------------START--------------  
  2.  
  3.  FAILRE - CAN not connect! Local: 127.0.0.1 remote: 10.8.1.50 port22  
  4.  FAILRE - CAN not connect! Local: 9.123.231.40 remote: 10.8.1.50 port22  
  5.  SUCCESS - connection established! Local: 10.0.50.189 remote: 10.8.1.50 port22  
  6.  Reachable local IP is found, it is 10.0.50.189  
  7.  
  8. --------------END-------------- 

 當(dāng)網(wǎng)絡(luò)環(huán)境中存在 IPv4 和 IPv6,即機(jī)器既有 IPv4 地址,又有 IPv6 地址的時(shí)候,我們可以對(duì)程序進(jìn)行一些優(yōu)化,比如

  • 由于 IPv4 和 IPv6 地址之間是無(wú)法互相訪問(wèn)的,因此僅需要判斷 IPv4 地址之間和 IPv6 地址之間的可達(dá)性。
  • 對(duì)于 IPv4 的換回地址可以不做判斷,對(duì)于 IPv6 的 Linklocal 地址也可以跳過(guò)測(cè)試
  • 根據(jù)實(shí)際的需要,我們可以?xún)?yōu)先考慮選擇使用 IPv4 或者 IPv6,提高判斷的效率

判斷本地地址和遠(yuǎn)程地址是否同為 IPv4 或者 IPv6

 
 
  1. // 判斷是 IPv4 還是 IPv6  
  2.  if(!((localInetAddr instanceofInet4Address) && (remoteInetAddr instanceofInet4Address)  
  3.  || (localInetAddr instanceofInet6Address) && (remoteInetAddr instanceofInet6Address))){  
  4.  // 本地和遠(yuǎn)程不是同時(shí)是 IPv4 或者 IPv6,跳過(guò)這種情況,不作檢測(cè) 
  5.  break;  
  6.  } 

跳過(guò)本地地址和 LinkLocal 地址

 
 
  1. if( localAddr.isLoopbackAddress() ||  
  2.      localAddr.isAnyLocalAddress() ||  
  3.      localAddr.isLinkLocalAddress() ){  
  4.      // 地址為本地環(huán)回地址,跳過(guò) 
  5.      break;  
  6.  } 

本文列舉集中典型的場(chǎng)景,介紹了通過(guò) Java 網(wǎng)絡(luò)編程接口判斷機(jī)器之間可達(dá)性的幾種方式。在實(shí)際應(yīng)用中,可以根據(jù)不同的需要選擇相應(yīng)的方法稍加修改即可。對(duì)于更加特殊的需求,還可以考慮通過(guò) JNI 的方法直接調(diào)用系統(tǒng) API 來(lái)實(shí)現(xiàn),能提供更加強(qiáng)大和靈活的功能,這里就不再贅述了。

文章來(lái)源:http://www.ibm.com/developerworks/cn/java/j-lo-connectiontest/


本文題目:使用Java測(cè)試網(wǎng)絡(luò)連通性的幾種方法
當(dāng)前路徑:http://m.5511xx.com/article/coopjoi.html