此版本的 Java远程获取MAC地址代码 与上一版本相比,多了连接超时

package com.thunisoft.shxt.support.macAddr;

import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.net.SocketException;
import java.net.SocketTimeoutException;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

/**
 * 主机A向主机B发送“UDP-NetBIOS-NS”询问包,即向主机B的137端口,发Query包来询问主机B的NetBIOS Names信息。
 * 其次,主机B接收到“UDP-NetBIOS-NS”询问包,
 * 假设主机B正确安装了NetBIOS服务........... 而且137端口开放,
 * 则主机B会向主机A发送一个“UDP-NetBIOS-NS”应答包,即发Answer包给主机A。
 * 并利用UDP(NetBIOS Name Service)来快速获取远程主机MAC地址的方法
 * @author lijt
 */

public class UdpGetClientMacAddr {
    
    private static Log log = LogFactory.getLog(UdpGetClientMacAddr.class);
    
    private String         remoteAddr;
    private int            remotePort = 137;
    private byte[]         buffer      = new byte[1024];
    private DatagramSocket ds          = null;

    public UdpGetClientMacAddr(String strAddr) throws Exception {
        remoteAddr = strAddr;
        ds = new DatagramSocket();
    }

    //发送数据包
    protected final DatagramPacket send(final byte[] bytes) throws IOException {
        DatagramPacket dp = new DatagramPacket(bytes, bytes.length, InetAddress.getByName(remoteAddr), remotePort);
        ds.send(dp);
        return dp;
    }
    //接收数据包
    protected final DatagramPacket receive() {
        DatagramPacket dp = new DatagramPacket(buffer, buffer.length);
        try {
            ds.setSoTimeout(3000);
            ds.receive(dp); 
        }catch(SocketTimeoutException  ex) {   
           log.info("接收数据超时...,不能获取客户端MAC地址");
          //  throw new SocketTimeoutException("连接超时");
        }  catch (SocketException e1) {
            log.error("发生Sorcket异常..."+e1.getMessage());
            e1.printStackTrace();
        }catch (IOException e2) {
            log.error("发生IO异常..."+e2.getMessage());
        }
        return dp;
    }

    // 询问包结构:
    // Transaction ID 两字节(16位) 0x00 0x00
    // Flags 两字节(16位) 0x00 0x10
    // Questions 两字节(16位) 0x00 0x01
    // AnswerRRs 两字节(16位) 0x00 0x00
    // AuthorityRRs 两字节(16位) 0x00 0x00
    // AdditionalRRs 两字节(16位) 0x00 0x00
    // Name:array [1..34] 0x20 0x43 0x4B 0x41(30个) 0x00 ;
    // Type:NBSTAT 两字节 0x00 0x21
    // Class:INET 两字节(16位)0x00 0x01
    protected byte[] getQueryCmd() throws Exception {
        byte[] t_ns = new byte[50];
        t_ns[0] = 0x00;
        t_ns[1] = 0x00;
        t_ns[2] = 0x00;
        t_ns[3] = 0x10;
        t_ns[4] = 0x00;
        t_ns[5] = 0x01;
        t_ns[6] = 0x00;
        t_ns[7] = 0x00;
        t_ns[8] = 0x00;
        t_ns[9] = 0x00;
        t_ns[10] = 0x00;
        t_ns[11] = 0x00;
        t_ns[12] = 0x20;
        t_ns[13] = 0x43;
        t_ns[14] = 0x4B;

        for (int i = 15; i < 45; i++) {
            t_ns[i] = 0x41;
        }

        t_ns[45] = 0x00;
        t_ns[46] = 0x00;
        t_ns[47] = 0x21;
        t_ns[48] = 0x00;
        t_ns[49] = 0x01;
        return t_ns;
    }

    // 表1 “UDP-NetBIOS-NS”应答包的结构及主要字段一览表
    // 序号 字段名 长度
    // 1 Transaction ID 两字节(16位)
    // 2 Flags 两字节(16位)
    // 3 Questions 两字节(16位)
    // 4 AnswerRRs 两字节(16位)
    // 5 AuthorityRRs 两字节(16位)
    // 6 AdditionalRRs 两字节(16位)
    // 7 Name<Workstation/Redirector> 34字节(272位)
    // 8 Type:NBSTAT 两字节(16位)
    // 9 Class:INET 两字节(16位)
    // 10 Time To Live 四字节(32位)
    // 11 Length 两字节(16位)
    // 12 Number of name 一个字节(8位)
    // NetBIOS Name Info 18×Number Of Name字节
    // Unit ID 6字节(48位

    protected final String getMacAddr(byte[] brevdata) throws Exception {
        // 获取计算机名
     //   System.out.println(new String(brevdata, 57, 18));
     //   System.out.println(new String(brevdata, 75, 18));
     //   System.out.println(new String(brevdata, 93, 18));
        int i = brevdata[56] * 18 + 56;
        String sAddr = "";
        StringBuffer sb = new StringBuffer(17);
        // 先从第56字节位置,读出Number Of Names(NetBIOS名字的个数,其中每个NetBIOS Names Info部分占18个字节)
        // 然后可计算出“Unit ID”字段的位置=56+Number Of Names×18,最后从该位置起连续读取6个字节,就是目的主机的MAC地址。
        for (int j = 1; j < 7; j++) {
            sAddr = Integer.toHexString(0xFF & brevdata[i + j]);
            if (sAddr.length() < 2) {
                sb.append(0);
            }
            sb.append(sAddr.toUpperCase());
            if (j < 6) sb.append('-');
        }
        return sb.toString();
    }

    public final void close() {
        try {
            ds.close();
        } catch (Exception ex) {
            ex.printStackTrace();
        }
    }

    /**
     * 获取远程主机的mac地址
     * @return
     * @throws Exception
     */
    public final String getRemoteMacAddr() throws Exception {
        byte[] bqcmd = getQueryCmd();
        this.send(bqcmd);
        DatagramPacket dp = receive();
        String smac ="";
        smac = getMacAddr(dp.getData());
        this.close();
        return smac;
    }
   public static void  main(String[] args) throws Exception
   {
       UdpGetClientMacAddr  add = new UdpGetClientMacAddr("172.16.26.52");
       System.out.println(add.getRemoteMacAddr());
   }

}

  

posted on 2015-03-16 09:19  zhangdong_ah  阅读(2688)  评论(0编辑  收藏  举报