现在我们的项目现在加入了身份验证,通过收集客户端的电脑的CPU编号、硬盘编号和网卡编号然后生成一个机器码,然后使用这个机器码作为身份标识,验证客户端的合法性,我们的项目是RCP架构,所以使用了SWT Extension这样的一个插件(国人开发,而且是BlogJava的会员),这样比较方便,获取CPU编号、硬盘编号倒是没有什么问题,获取网卡信息的时候有些电脑居然采集不到信息,采集方式是这样的:
if (macs == null || macs.length == 0 ) {
macs = Extension.GetMACAddress( 1 );
}
if (macs == null || macs.length == 0 ) {
macs = Extension.GetMACAddress( 2 );
}
if (macs == null || macs.length == 0 ) {
macs = Extension.GetMACAddress( 3 );
}
StringBuffer stringBuffer = new StringBuffer();
for ( int i = 0 ; i < macs.length; i ++ ) {
stringBuffer.append(getHexString(macs[i], 2 )
if (i != macs.length - 1 )
stringBuffer.append( " - " );
}
System.out.println(stringBuffer.toString.toUpperCase()););
注意通过Extension.GetMACAddress()方法得到的网卡地址是一个十进制的数组,需要将他转换成十六进制的,这样才能和操作系统现在的信息一致。
这种采集方式不是万能,估计和操作环境有关系,我们这个项目的按照客户端大概在4000台电脑左右,大部分安装的Windows 2000操作系统、Java运行环境是jre 6.0,通过分析发现有些电脑是双网卡,我见得最多的一个电脑居然配置了9块网卡(包括虚拟的和禁用掉的),在这种情况下SWT Extension 是无法正常工作了,网络上也提供了其他的办法比如使用
Process p = Runtime.getRuntime().exec(command);
BufferedReader br = new BufferedReader( new InputStreamReader(p.getInputStream()));
String line;
while ((line = br.readLine()) != null ) {
System.out.println( " line: " + line);
if (line.indexOf( " Physical Address " ) > 0 ) {
int index = line.indexOf( " : " );
index += 2 ;
address = line.substring(index);
break ;
}
}
br.close();
根据返回的输入流从里面解析信息,通过使用发现这种方式也不太管用!。这里重点介绍第三种方式:通过JDK 6.0中的方法来获取:
try {
Enumeration < NetworkInterface > el = NetworkInterface.getNetworkInterfaces();
while (el.hasMoreElements()) {
byte [] mac = el.nextElement().getHardwareAddress();
if (mac == null )
continue ;
StringBuilder builder = new StringBuilder();
for ( byte b : mac) {
builder.append(hexByte(b));
builder.append( " - " );
}
builder.deleteCharAt(builder.length() - 1 );
return builder.toString();
}
}
catch (Exception exception) {
exception.printStackTrace();
}
return null ;
}
这是JDK 6.0的新特性,可以支持多网卡配置,你只需要判断一下就行
我的做法是将这三种方法同时使用:如果第一种方式获取不到,换成第二种,第二种方法也无效的话,我就使用第三种,如果第三种方式仍然无效的话,只能建议客户重装系统了:你还是升级到XP系统吧