在Java开发中,我们经常会遇到需要将MAC地址转换为长整型数值的场景,比如做设备唯一标识存储、网络数据包处理等。MAC地址本身是48位的十六进制数值,而Java的long类型是64位,完全能够容纳转换后的结果。下面我们就一步步讲解具体的实现方法。

转换核心思路
MAC地址的标准格式通常是XX:XX:XX:XX:XX:XX,每组是两个十六进制字符,总共6组。转换的核心逻辑就是先去掉分隔符,把整个MAC地址拼接成一个12位的十六进制字符串,再把这个字符串转换为long类型的数值即可。不过需要注意,十六进制字符串转long的时候要处理无符号的情况,避免高位被当成负数。
实现步骤详解
1. 校验并清洗MAC地址
首先需要处理MAC地址的分隔符,常见的分隔符有冒号、横杠,甚至可能没有分隔符,我们需要先统一清洗掉这些非十六进制字符,同时校验格式是否合法。
2. 十六进制字符串转长整型
清洗后的12位十六进制字符串,需要使用Long.parseLong方法,指定进制为16,同时因为MAC地址是无符号的48位数值,转换后如果高位为1,Java的long会显示为负数,我们可以根据实际需求决定是否做无符号处理。
完整代码示例
下面是完整的转换工具类代码,包含格式校验和转换逻辑:
import java.util.regex.Pattern;
public class MacToLongUtil {
// 匹配MAC地址的正则,支持冒号、横杠分隔或无分隔符
private static final Pattern MAC_PATTERN = Pattern.compile("^([0-9A-Fa-f]{2}[:-]?){5}[0-9A-Fa-f]{2}$");
/**
* 将MAC地址转换为长整型数值
* @param macAddress 原始MAC地址,支持XX:XX:XX:XX:XX:XX、XX-XX-XX-XX-XX-XX或XXXXXXXXXXXX格式
* @return 转换后的长整型数值
* @throws IllegalArgumentException 如果MAC地址格式不合法
*/
public static long macToLong(String macAddress) {
if (macAddress == null || macAddress.trim().isEmpty()) {
throw new IllegalArgumentException("MAC地址不能为空");
}
// 校验MAC地址格式
if (!MAC_PATTERN.matcher(macAddress).matches()) {
throw new IllegalArgumentException("非法的MAC地址格式:" + macAddress);
}
// 去掉所有分隔符,只保留十六进制字符
String cleanMac = macAddress.replaceAll("[:-]", "");
// 十六进制字符串转long,16进制
return Long.parseLong(cleanMac, 16);
}
/**
* 将长整型数值转换回MAC地址(可选,用于验证转换结果)
* @param macLong 长整型MAC数值
* @return 标准冒号分隔的MAC地址
*/
public static String longToMac(long macLong) {
// 转成16进制,补全前导0到12位
String hex = String.format("%012x", macLong);
// 每两位加一个冒号分隔
return hex.replaceAll("(.{2})(?=.)", "$1:");
}
// 测试示例
public static void main(String[] args) {
String mac = "AA:BB:CC:DD:EE:FF";
long macLong = macToLong(mac);
System.out.println("MAC地址:" + mac);
System.out.println("转换后的长整型:" + macLong);
System.out.println("转回MAC地址验证:" + longToMac(macLong));
}
}注意事项
- MAC地址转换后的长整型如果高位为1,会出现负数,这是因为Java的long是有符号的,如果需要无符号表示,可以用Long.toUnsignedString方法处理。
- 转换前一定要做格式校验,避免非法字符串导致转换异常。
- 如果MAC地址是小写字母,不影响转换结果,因为Long.parseLong方法对大小写十六进制字符都支持。
常见问题解答
转换后数值是负数怎么办
这是正常的,因为48位MAC地址的最高位如果是1,转换为64位有符号long时就会被识别为负数。如果需要无符号的字符串表示,可以调用Long.toUnsignedString(macLong, 16)得到无符号的十六进制字符串。
支持不带分隔符的MAC地址吗
上面的代码已经支持,正则校验部分允许没有分隔符的12位十六进制字符串,清洗步骤也会自动处理这种情况。