最近工作中需要在网络中传递mac地址,需要将mac地址转换为六个字节,网上找了找资料,基本的思路就是用根据冒号将mac地址分为六段, 每段看做一个十六进制数,转化为十进制数后,分配赋值给六字节数组的每一位。
后来在查看库函数的时候,发现了strtoul函数,发现这个函数正好可以用来实现这个功能。
实现函数如下:
int mac_str_to_bin( char *str, unsigned char *mac)
{
int i;
char *s, *e;
if ((mac == NULL) || (str == NULL))
{
return -1;
}
s = (char *) str;
for (i = 0; i < 6; ++i)
{
mac[i] = s ? strtoul (s, &e, 16) : 0;
if (s)
s = (*e) ? e + 1 : e;
}
return 0;
}
测试程序如下:
int main(void)
{
char *p = "11:22:33:44:55:66";
unsigned char dst[6];
mac_str_to_bin(p,dst);
printf("mac:%2.2x:%2.2x:%2.2x:%2.2x:%2.2x:%2.2x\n",
dst[0], dst[1], dst[2], dst[3], dst[4], dst[5]);
return 0;
}
打印结果为:mac:11:22:33:44:55:66
unsigned long int strtoul(const char *nptr, char **endptr, int base);
先介绍下strtoul函数,它会从一个数字字符或是正负号字符开始,直到遇见非数字字符或字符串,停止读取,将已读取到的数字字符串,按照base指定的进制方式,转化为长整型数,nptr会指向停止时读取的位置,即读取过程中遇到的非数字字符或字符串的位置。这个地方也是endptr指针也是通过二级指针出参来达到目的的。
上面的实现函数中,主要的实现都在for循环里面,第一句mac[i] = s ? strtoul (s, &e, 16) : 0; 先判断s是否为空,如果上一次已经将整个mac都解析完了的话,s会置为空;如果s不为空,即表示mac地址解析还没有完成,通过调用strtoul(),解析mac地址中的一段。第二句 if (s) s = (*e) ? e + 1 : e; 如果s不为空,则表示解析没有完成,s当前为上次解析时遇见的非数字字符或字符串的地址,即分隔符冒号‘:’的位置,这句就是为了去除掉这个冒号。
其实在linux源码中,这个的时候也是使用的本文章最开始提出的那种思路,只不过封装进函数库中了。既然库函数中都有api接口了,就不用我们自己写了。
下面附上本函数的linux源码:
unsigned long long int strtoull(const char *ptr, char **end, int base)
{
unsigned long long ret = 0;
if (base > 36)
goto out;
while (*ptr) {
int digit;
if (*ptr >= '0' && *ptr <= '9' && *ptr < '0' + base)
digit = *ptr - '0';
else if (*ptr >= 'A' && *ptr < 'A' + base - 10)
digit = *ptr - 'A' + 10;
else if (*ptr >= 'a' && *ptr < 'a' + base - 10)
digit = *ptr - 'a' + 10;
else
break;
ret *= base;
ret += digit;
ptr++;
}
out:
if (end)
*end = (char *)ptr;
return ret;
}
这个代码是由Scott Wood编写的。