這篇文章主要講解了“Java網(wǎng)絡(luò)連通性如何測試”,文中的講解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“Java網(wǎng)絡(luò)連通性如何測試”吧!
創(chuàng)新互聯(lián)建站堅信:善待客戶,將會成為終身客戶。我們能堅持多年,是因為我們一直可值得信賴。我們從不忽悠初訪客戶,我們用心做好本職工作,不忘初心,方得始終。10多年網(wǎng)站建設(shè)經(jīng)驗創(chuàng)新互聯(lián)建站是成都老牌網(wǎng)站營銷服務(wù)商,為您提供網(wǎng)站設(shè)計、成都做網(wǎng)站、網(wǎng)站設(shè)計、H5技術(shù)、網(wǎng)站制作、高端網(wǎng)站設(shè)計、微信小程序服務(wù),給眾多知名企業(yè)提供過好品質(zhì)的建站服務(wù)。
一般情況下,我們僅僅需要判斷從一臺機(jī)器是否可以訪問(Ping)到另一臺機(jī)器,此時,可以簡單的使用 Java 類庫中 java.net.InetAddress 類來實現(xiàn),這個類提供了兩個方法探測遠(yuǎn)程機(jī)器是否可達(dá)
boolean isReachable(int timeout) // 測試地址是否可達(dá) boolean isReachable(NetworkInterface netif, int ttl, int timeout) // 測試地址是否可達(dá).
簡單說來,上述方法就是通過遠(yuǎn)端機(jī)器的 IP 地址構(gòu)造 InetAddress 對象,然后調(diào)用其 isReachable 方法,測試調(diào)用機(jī)器和遠(yuǎn)端機(jī)器的網(wǎng)絡(luò)可達(dá)性。注意到遠(yuǎn)端機(jī)器可能有多個 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"); EnumerationnetInterfaces = 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 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 的用法,可以不指定任何接口來判斷遠(yuǎn)端網(wǎng)絡(luò)的可達(dá)性,但這不能區(qū)分出數(shù)據(jù)包是從那個網(wǎng)絡(luò)接口發(fā)出去的 ( 如果本地有多個網(wǎng)絡(luò)接口的話 );而高級版本的 isReachable 則可以指定從本地的哪個網(wǎng)絡(luò)接口測試,這樣可以準(zhǔn)確的知道遠(yuǎn)端網(wǎng)絡(luò)可以連通本地的哪個網(wǎng)絡(luò)接口。
但是,Java 本身沒有提供任何方法來判斷本地的哪個 IP 地址可以連通遠(yuǎn)端網(wǎng)絡(luò),Java 網(wǎng)絡(luò)編程接口也沒有提供方法來訪問 ICMP 協(xié)議數(shù)據(jù)包,因而通過 ICMP 的網(wǎng)絡(luò)不可達(dá)數(shù)據(jù)包實現(xiàn)這一點也是不可能的 ( 當(dāng)然可以用 JNI 來實現(xiàn),但就和系統(tǒng)平臺相關(guān)了 ), 此時可以考慮本文下一節(jié)提出的方法。
在某些情況下,我們可能要確定本地的哪個網(wǎng)絡(luò)地址可以連通遠(yuǎn)程網(wǎng)絡(luò),以便遠(yuǎn)程網(wǎng)絡(luò)可以回連到本地使用某些服務(wù)或發(fā)出某些通知。一個典型的應(yīng)用場景 是,本地啟動了文件傳輸服務(wù) ( 如 FTP),需要將本地的某個 IP 地址發(fā)送到遠(yuǎn)端機(jī)器,以便遠(yuǎn)端機(jī)器可以通過該地址下載文件;或者遠(yuǎn)端機(jī)器提供某些服務(wù),在某些事件發(fā)生時通知注冊了獲取這些事件的機(jī)器 ( 常見于系統(tǒng)管理領(lǐng)域 ),因而在注冊時需要提供本地的某個可達(dá) ( 從遠(yuǎn)端 ) 地址。
雖然我們可以用 InetAddress.isReachabl 方法判斷出本地的哪個網(wǎng)絡(luò)接口可連通遠(yuǎn)程玩過,但是由于單個網(wǎng)絡(luò)接口是可以配置多個 IP 地址的,因而在此并不合適。我們可以使用 Socket 建立可能的 TCP 連接,進(jìn)而判斷某個本地 IP 地址是否可達(dá)遠(yuǎn)程網(wǎng)絡(luò)。我們使用 java.net.Socket 類中的 connect 方法
void connect(SocketAddress endpoint, int timeout) //使用Socket連接服務(wù)器,指定超時的時間
這種方法需要遠(yuǎn)程的某個端口,該端口可以是任何基于 TCP 協(xié)議的開放服務(wù)的端口(如一般都會開放的 ECHO 服務(wù)端口 7, Linux 的 SSH 服務(wù)端口 22 等)。實際上,建立的 TCP 連接被協(xié)議棧放置在連接隊列,進(jìn)而分發(fā)到真正處理數(shù)據(jù)的各個應(yīng)用服務(wù),由于 UDP 沒有連接的過程,因而基于 UDP 的服務(wù)(如 SNMP)無法在此方法中應(yīng)用。
具體過程是,枚舉本地的每個網(wǎng)絡(luò)地址,建立本地 Socket,在某個端口上嘗試連接遠(yuǎn)程地址,如果可以連接上,則說明該本地地址可達(dá)遠(yuǎn)程網(wǎng)絡(luò)。
void printReachableIP(InetAddress remoteAddr, int port){ String retIP = null; EnumerationnetInterfaces; try{ netInterfaces = NetworkInterface.getNetworkInterfaces(); while(netInterfaces.hasMoreElements()) { NetworkInterface ni = netInterfaces.nextElement(); Enumeration 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(); // 端口號設(shè)置為 0 表示在本地挑選一個可用端口進(jìn)行連接 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; }
運行結(jié)果
--------------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--------------
當(dāng)網(wǎng)絡(luò)環(huán)境中存在 IPv4 和 IPv6,即機(jī)器既有 IPv4 地址,又有 IPv6 地址的時候,我們可以對程序進(jìn)行一些優(yōu)化,比如
由于 IPv4 和 IPv6 地址之間是無法互相訪問的,因此僅需要判斷 IPv4 地址之間和 IPv6 地址之間的可達(dá)性。
對于 IPv4 的換回地址可以不做判斷,對于 IPv6 的 Linklocal 地址也可以跳過測試
根據(jù)實際的需要,我們可以優(yōu)先考慮選擇使用 IPv4 或者 IPv6,提高判斷的效率
判斷本地地址和遠(yuǎn)程地址是否同為 IPv4 或者 IPv6
// 判斷是 IPv4 還是 IPv6 if(!((localInetAddr instanceofInet4Address) && (remoteInetAddr instanceofInet4Address) || (localInetAddr instanceofInet6Address) && (remoteInetAddr instanceofInet6Address))){ // 本地和遠(yuǎn)程不是同時是 IPv4 或者 IPv6,跳過這種情況,不作檢測 break; }
跳過本地地址和 LinkLocal 地址
if( localAddr.isLoopbackAddress() || localAddr.isAnyLocalAddress() || localAddr.isLinkLocalAddress() ){ // 地址為本地環(huán)回地址,跳過 break; }
感謝各位的閱讀,以上就是“Java網(wǎng)絡(luò)連通性如何測試”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對Java網(wǎng)絡(luò)連通性如何測試這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是創(chuàng)新互聯(lián),小編將為大家推送更多相關(guān)知識點的文章,歡迎關(guān)注!