单网卡与多网卡环境下获取MAC地址

一、场景引入

       前一段时间写的一个服务需要获取服务所在的服务器mac地址当做唯一身份标识去更新当前服务器的信息。因为生产环境的系统是centos,故按照linux的系统去获取mac地址。java在JDK1.6以后 java.net.NetworkInterface提供了

整的方法用于获取网络设备信息,调用 NetworkInterface.getNetworkInterfaces()可以返回所有网络设备,调用NetworkInterface.getHardwareAddress()就可以获取指定网卡的MAC。但是有一种情况需要考虑,也就是要提前知道服

务所在的机器是单网卡还是多网卡,为什么这么说呢?因为我在写完服务之后放到测试环境一切正常,部署到生产环境的时候出问题了,获取mac地址异常,后来发现生产环境是双网卡。

二、解决方法

     在此说一下单网卡如何获取mac地址,windows系统和linux系统获取的方法也不一样,如果提前不知道系统的情况下还需要对系统进行区分(这种情况应该很少会有),故现在只针对linux系统获取mac地址的方法,获取后转换为大写,并用“-”连接

/**
* 获取本机mac地址
* @return
* @throws Exception
*/
public static String getLinuxMacAdress() throws UnknownHostException, SocketException{
InetAddress ia = InetAddress.getLocalHost();
byte[] mac = NetworkInterface.getByInetAddress(ia).getHardwareAddress();
StringBuilder sb = new StringBuilder("");
for(int i=0; i<mac.length; i++) {
if(i!=0) {
sb.append("-");
}
//字节转换为整数
int temp = mac[i]&0xff;
String str = Integer.toHexString(temp);
if(str.length()==1) {
sb.append("0"+str);
}else {
sb.append(str);
}
}
return sb.toString().toUpperCase();
}

       针对多网卡环境,可以有两种方法去解决,以上单网卡mac地址的获取操作在多网卡环境下会抛异常,所以第一种方式可以在抛异常的时候读取配置文件中提前配置的mac地址,也就是说,在部署的时候就人工将mac地址写到配置文件中,一旦mac读取失败就去读取配置文件。这么做看起来没有实际解决获取mac的实际问题,但是可以解决一部分场景下的需求了

       第二种方法就是真正的去获取多网卡环境下的mac地址,仍然写成工具类的形式,直接上代码:

    

public class NetworkUtil {
    public static enum Radix{
    /** 二进制 */BIN(2),
    /** 十进制 */DEC(10),
    /** 十六进制 */HEX(16);
   final int value;
   Radix(int radix){
   this.value = radix;
}
}
public static enum Filter implements Predicate<NetworkInterface>{
/** 过滤器: 所有网卡 */ALL,
/** 过滤器: 在线设备,see also {@link NetworkInterface#isUp()} */UP,
/** 过滤器: 虚拟接口,see also {@link NetworkInterface#isVirtual()} */VIRTUAL,
/** 过滤器:LOOPBACK, see also {@link NetworkInterface#isLoopback()} */LOOPBACK,
/** 过滤器:物理网卡 */PHYICAL_ONLY;

@Override
public boolean apply(NetworkInterface input) {
if(null == input ){
return false;
}
try{
byte[] hardwareAddress;
switch(this){
case UP:
return input.isUp();
case VIRTUAL:
return input.isVirtual();
case LOOPBACK:
return input.isLoopback();
case PHYICAL_ONLY :
hardwareAddress = input.getHardwareAddress();
return null != hardwareAddress
&& hardwareAddress.length > 0
&& !input.isVirtual()
&& !isVMMac(hardwareAddress);
case ALL:
default :
return true;
}
} catch (SocketException e) {
throw new RuntimeException(e);
}
}
}
/**
* 根据过滤器{@code filters}指定的条件(AND)返回网卡设备对象
* @param filters
* @return
*/
@SafeVarargs
@SuppressWarnings("unchecked")
public static Set<NetworkInterface> getNICs(Predicate<NetworkInterface> ...filters) {
if(null == filters){
filters = new Predicate[]{Filter.ALL};
}
try {
Iterator<NetworkInterface> filtered = Iterators.filter(
Iterators.forEnumeration(NetworkInterface.getNetworkInterfaces()),
Predicates.and(filters));
return ImmutableSet.copyOf(filtered);
} catch (SocketException e) {
throw new RuntimeException(e);
}
}
/**
* 返回所有物理网卡
* @return
*/
public static Set<NetworkInterface> getPhysicalNICs() {
return getNICs(Filter.PHYICAL_ONLY,Filter.UP);
}
/**
* 将{@code byte[]} 转换为{@code radix}指定格式的字符串
*
* @param source
* @param separator 分隔符
* @param radix 进制基数
* @return {@code source}为{@code null}时返回空字符串
*/
public static final String format(byte[] source,String separator, final Radix radix) {
if (null == source){
return "";
}
if(null == separator){
separator = "";
}
List<String> hex = Lists.transform(Bytes.asList(source),new Function<Byte,String>(){
@Override
public String apply(Byte input) {
return String.copyValueOf(new char[]{
Character.forDigit((input & 240) >> 4, radix.value),
Character.forDigit(input & 15, radix.value)
});
}});
return Joiner.on(separator).join(hex);
}
/**
* MAC地址格式(16进制)格式化{@code source}指定的字节数组
* @see #format(byte[], String, int)
*/
public static final String formatMac(byte[] source,String separator) {
return format(source,separator,Radix.HEX);
}
/**
* 以IP地址格式(点分位)格式化{@code source}指定的字节数组<br>
* @see #format(byte[], String, int)
*/
public static final String formatIp(byte[] source) {
return format(source,".",Radix.DEC);
}
/**
* 返回指定{@code address}绑定的网卡的物理地址(MAC)
* @param address
* @return 指定的{@code address}没有绑定在任何网卡上返回{@code null}
* @see {@link NetworkInterface#getByInetAddress(InetAddress)}
* @see {@link NetworkInterface#getHardwareAddress()}
*/
public static byte[] getMacAddress(InetAddress address) {
  try {
       NetworkInterface nic = NetworkInterface.getByInetAddress(address);
       return null == nic ? null : nic.getHardwareAddress();
  } catch (SocketException e) {
        throw new RuntimeException(e);
  }
}
/**
* @param nic 网卡对象
* @param separator 格式化分隔符
* @return 表示MAC地址的字符串
*/
public static String getMacAddress(NetworkInterface nic,String separator) {
try {
return format(nic.getHardwareAddress(),separator, Radix.HEX);
} catch (SocketException e) {
throw new RuntimeException(e);
}
}
/**
* 参见 {@link #getMacAddress(InetAddress)}
* @param address
* @param separator 格式化分隔符
* @return 表示MAC地址的字符串
*/
public static String getMacAddress(InetAddress address,String separator) {
      return format(getMacAddress(address),separator, Radix.HEX);
}
private static byte invalidMacs[][] = {
    {0x00, 0x05, 0x69}, // VMWare
    {0x00, 0x1C, 0x14}, // VMWare
    {0x00, 0x0C, 0x29}, // VMWare
    {0x00, 0x50, 0x56}, // VMWare
    {0x08, 0x00, 0x27}, // Virtualbox
    {0x0A, 0x00, 0x27}, // Virtualbox
    {0x00, 0x03, (byte)0xFF}, // Virtual-PC
    {0x00, 0x15, 0x5D} // Hyper-V
};
private static boolean isVMMac(byte[] mac) {
    if(null == mac) {
    return false;
}

for (byte[] invalid: invalidMacs){
      if (invalid[0] == mac[0] && invalid[1] == mac[1] && invalid[2] == mac[2]) {
       return true;
      }
  }

   return false;
 }

}

    

 

 

       

posted @ 2018-12-05 16:34  技术驱动产品  阅读(2653)  评论(0编辑  收藏  举报