Chinaunix首页 | 论坛 | 博客
  • 博客访问: 7941130
  • 博文数量: 701
  • 博客积分: 2150
  • 博客等级: 上尉
  • 技术积分: 13233
  • 用 户 组: 普通用户
  • 注册时间: 2011-06-29 16:28
个人简介

天行健,君子以自强不息!

文章分类

全部博文(701)

文章存档

2019年(2)

2018年(12)

2017年(76)

2016年(120)

2015年(178)

2014年(129)

2013年(123)

2012年(61)

分类: Java

2016-06-15 14:34:31

原文链接
http://blog.sina.com.cn/s/blog_616e189f0100rx2i.html

第三章:
java网络编程

         使用java进行网络程序的开发,可以说是一件令人愉悦的事情,对于用惯了C++网络接口编程的人来说,当他们首次使用Java开发网络应用程序,会发现java开发网络应用是如此的简单,甚至仅用几分钟时间,您就可以学会这种网络编程技术。本章主要对java网络编程相关的知识进行探讨,首先探讨java对于网卡信息获取方式,之后介绍如何使用java编写基于网络的客户端和服务端程序。最后探讨一些java网络相关的高级主题。

         3.1 java获取网络通信信息

         java为了方便网络编程,提供了表示IP地址的类、表示网络接口(这个接口是指网卡)的类,表示网络连接接口的类。下面主要探讨这些基本的类,以及如何使用这些类来获取网络信息。Graphic3-1给出了与网络通信信息相关的类之间的关系。

         Graphic3-1 网络通信信息相关的类

java 网络编程 -- IP地址的表示与网络接口信息的获取(InetAddress和NetworkInterface)

 

         3.1.1 InetAddress

         InetAddress用来表示一个IP地址。可以通过主机名来配置InetAddress,此时InetAddress将会通过主机配置的DNS服务器解析出对应域名的IP地址。对于任何的主机名,InetAddress也可以解析出其对应的IP地址。InetAddress拥有两个子类,分别是Inet4AddressInet6Address,在大多说情况下都无需直接处理这些子类,因为InetAddress抽象类已经覆盖了大多数必需的功能。Graphic3-2给出了InetAddress的公有操作。从Graphic3-2从可以看出,InetAddress类没有公有的构造方法,要想获取InetAddress的实例必需使用InetAddress内部的一些静态方法(这是设计模式中的简单工厂方法)。

Demo3-1给出了使用InetAddress的基本用法。示例中先使用InetAddressgetLocalHost方法获取了本机的IP地址,然后输出,使用getByName方法获得了google服务器的一个IP地址,由于google服务器有很多服务主机,当前获取的是哪一个服务的主机的IP地址并不确定,实际应用中,这是一个随机的值,这与DNS服务器有直接关系。使用了getByAllName方法可以获取google的所有服务主机名。最后使用了isReachable方法来判断一台主机的可达性。

需要说明的是,使用getLocalHost方法未必真的能够获取您想要的本机的IP地址(有人说,在windowsgetLocalHost可以正确执行,获取到本机的IPV4地址,而在Linux上调用getLocalHost返回的是一个127.0.0.1ip地址)。现在假想有一台主机有多张网卡,而且有多个IP地址,甚至这台主机即配置了IPV4又配置了IPV6,调用getLocalHost将会返回哪一个IP地址?很明显getLocalHost只能返回一个IP地址,因此就限定了getLocalHost方法并不是获取本机IP地址的最好方式。

Demo3-2给出了一台Linux系统一张网卡上配置IPV4IPV6后,调用getLocalHost后的执行结果。从Configuration3-2中可以看出,在这台linux上配置了IPV4IPV6,调用getLocalHost后,获取的是一个回环测试地址。

此外,使用InetAddressisReachable时需要特别小心,因为默认情况下,isReachable使用ICMP报文来探测远程主机可达性(不一定使用ICMP,这与具体的实现有关),有很多服务器或者网络管理员禁用ping功能(ICMP),所以使用isReachable不可达,并不证明远程主机不存在。

 

Graphic3-2  InetAddress拥有的公有操作。

java 网络编程 -- IP地址的表示与网络接口信息的获取(InetAddress和NetworkInterface)

 

Demo 3-1 InetAddress示例代码

package com.upc.upcgrid.guan.network.chapter03;

 

import java.io.IOException;

import java.net.InetAddress;

 

public class InetAddressTest {

    public static void main(String[] args) throws IOException {

       InetAddress addr = InetAddress.getLocalHost();//获取本机ip

       System.out.println("local host : "+addr);    

      

       //获取指定服务的一个主机IP

       addr = InetAddress.getByName("google.com");

       System.out.println("google : " + addr);      

      

       //获取指定服务的所有主机IP

       InetAddress[] addrs = InetAddress.getAllByName("google.com");

       for(int i = 0 ;i < addrs.length ;i++)

           System.out.println("google : "+addrs[i] + "  number : " i);     

      

       //获取远程主机可达性

       System.out.println(InetAddress.getByName("localhost").isReachable(1000));   

    }

}

Demo3-1执行结果:

local host : PC2011022509cjh/202.194.158.128

google : google.com/74.125.71.99

google : google.com/74.125.71.99  number : 0

google : google.com/74.125.71.103  number : 1

google : google.com/74.125.71.104  number : 2

google : google.com/74.125.71.105  number : 3

google : google.com/74.125.71.106  number : 4

google : google.com/74.125.71.147  number : 5

true

demo3-1的配置:

configuration 3-1windows主机的配置

C:\Documents and Settings\Administrator>ipconfig

Windows IP Configuration

Ethernet adapter 本地连接:

        Connection-specific DNS Suffix  . :

        IP Address. . . . . . . . . . . . : 202.194.158.128

        Subnet Mask . . . . . . . . . . . : 255.255.255.0

        Default Gateway . . . . . . . . . : 202.194.158.253

 

Demo3-2 linux 执行的代码:

       System.out.println(InetAddress.getLocalHost());

Demo3-2 的输出结果:

Localhost.localdomain/127.0.0.1

Demo3-2 的配置环境

Configuration 3-2 linux主机的配置

[root@localhost ~]# ifconfig

eth0      Link encap:Ethernet  HWaddr 00:E0:4C:3D:C2:73 

          inet addr:202.194.158.184  Bcast:202.194.158.255  Mask:255.255.255.0

          inet6 addr: 2001:da8:7006:8003:2e0:4cff:fe3d:c273/64 Scope:Global

          inet6 addr: fe80::2e0:4cff:fe3d:c273/64 Scope:Link

          UP BROADCAST RUNNING MULTICAST  MTU:1500  Metric:1

          RX packets:251210956 errors:1 dropped:28 overruns:1 frame:0

          TX packets:116140638 errors:0 dropped:0 overruns:0 carrier:0

          collisions:0 txqueuelen:1000

          RX bytes:16023749113 (14.9 GiB)  TX bytes:34386034544 (32.0 GiB)

          Interrupt:217 Base address:0x2000

 

lo        Link encap:Local Loopback 

          inet addr:127.0.0.1  Mask:255.0.0.0

          inet6 addr: ::1/128 Scope:Host

          UP LOOPBACK RUNNING  MTU:16436  Metric:1

          RX packets:8274 errors:0 dropped:0 overruns:0 frame:0

          TX packets:8274 errors:0 dropped:0 overruns:0 carrier:0

          collisions:0 txqueuelen:0

          RX bytes:8395613 (8.0 MiB)  TX bytes:8395613 (8.0 MiB)

3.1.2 NetworkInterface

         可以使用NetworkInterface类提供的方法获取本地计算机网络接口相关的信息。尽管InetAddress类提获取IP地址的方法,但是要想获取本机的网络接口的详细信息,还需要依赖NetworkInterface接口中的方法。通过NetworkInterface接口,可以获取本机配置的网络接口的名字,IP列表(包括IPV4IPV6),网卡地址,最大传输单元(MTU),接口状态(开启/关闭)、接口网络类型(点对点网络、回环网络)等信息。Graphic3-3列出来NetworkInterface中可用的公有方法。从Graphic3-3可以看出,NetworkInterface没有可见的构造方法,NetworkInterface内部也是使用静态方法来构造NetworkInterface实例的(简单工厂方法)。此外,每个NetworkInterface内部可以包含多个NetworkInterface对象,这就构成了一种树结构(合成模式),而且每个NetworkInterface内部保持一个指向父NetworkInterface的引用。

         Demo3-3中给出了一个使用NetworkInterface获取本机eth0网络接口IPV4地址的一个示例程序,首先通过NetworkInterfacegetByName静态方法,获取一个关于eth0NetworkInterface实例,之后调用NetworkInterfacegetInetAddresses获取配置在eth0上的所有IP地址(这时有可能获取到多个IP,因为一个网络接口可以配置多个IP),通过遍历获取的InetAddress,判断如果是IPV4,则将这个IP地址输出出来。需要提示的是,一般每台计算机都会在别名为eth0的网络接口上配置公网的IPV4IPV6的地址,因此,直接使用eth0,即可获取配置的IP地址信息,此外还有一个别名为lo的网络接口,用来进行回环测试,其配置的IPV4一般为127.0.0.1Demo3-3中给出的例子解决了InetAddress调用getLocalHost方法无法正确获取本机IP地址的问题。为了保守起见(防止有的计算机没有为网络端口起名为eth0),demo3-4给出了一个更一般的方法。在Demo3-4中,先通过NetworkInterface的静态方法getNetworkInterfaces获取本机的所有网络接口的父接口,之后遍历这个父接口获取下面的子接口,(在Configuration1Configuration2中都配置了两个子接口,一个名字为eth0,一个名字为lo),之后遍历子接口的IP地址,输出结果。并且在输出中分辨出是IPV4的地址,还是IPV6的地址。

Graphic3-3 NetworkInterface


java 网络编程 -- IP地址的表示与网络接口信息的获取(InetAddress和NetworkInterface)

 

Demo3-3 获取本机IPV4的地址

package com.upc.upcgrid.guan.network.chapter03;

 

import java.net.InetAddress;

import java.net.NetworkInterface;

import java.net.SocketException;

import java.util.Enumeration;

 

public class NetworkInterfaceTest {

    public static void main(String[] args) throws SocketException {

       NetworkInterface ni = NetworkInterface.getByName("eth0");

       Enumeration ias = ni.getInetAddresses();

       for(;ias.hasMoreElements();)

       {

           InetAddress ia = ias.nextElement();

           if(ia instanceof InetAddress)

              System.out.println(ia);

       }

    }

}

Demo3-3的执行结构

Configuration3-1下:

/202.194.158.128

Configuration3-2下:

/202.194.158.184

Demo3-4 测试本机的IP地址

package com.upc.upcgrid.guan.network.chapter03;

 

import java.net.Inet4Address;

import java.net.Inet6Address;

import java.net.InetAddress;

import java.net.NetworkInterface;

import java.net.SocketException;

import java.util.Enumeration;

 

public class NetworkInterfaceTest2 {

    public static void main(String[] args) throws SocketException {

       Enumeration nis = NetworkInterface.getNetworkInterfaces();

       for(;nis.hasMoreElements();)

       {

           NetworkInterface ni = nis.nextElement();

           Enumeration ias = ni.getInetAddresses();

           for(;ias.hasMoreElements();)

           {

              InetAddress ia = ias.nextElement();

              if(ia instanceof Inet4Address && !ia.getHostAddress().equals("127.0.0.1"))

              {

                  System.out.println("ip v4 : "+ia);

              }else if(ia instanceof Inet6Address && !ia.equals(""))

              {

                  System.out.println("ip v6 : "+ ia);

              }

           }

          

       }

    }

}

Demo3-4的执行结果:

Configuration3-1 下的执行结果:

ip v4 : /202.194.158.128

Configuration3-2下的执行结果:

ip v6 : /fe80:0:0:0:2e0:4cff:fe3d:c273%2

ip v6 : /2001:da8:7006:8003:2e0:4cff:fe3d:c273%2

ip v4 : /202.194.158.184

ip v6 : /0:0:0:0:0:0:0:1%1

3.1.3 简单总结

         InetAddress用于表示一个IP地址,其两个子类Inet4AddressInet6Address分别表示IPV4IPV6两种类型的地址。使用InetAddressgetByName方法可以获取远程服务的IP地址(通过DNS服务获取的),使用getAllByName方法可以获取远程指定服务的所有服务主机的IP地址。使用getLocalHost方法可以获取本地IP地址,但是这种方式并不可靠,当出现多张网卡,或一个网络接口配置了多个IP,或者不同的操作系统类型,都不能保证能够获得想要的IP

         NetworkInterface可以获取本机网络接口的相关信息,包括硬件地址,MTU,所有的IP地址等信息,需要获取本机IP时,最好使用NetworkInterface对配置的IP地址进行筛选。

阅读(2841) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~