最近在完成计算机网络课程设计的过程中,需要将mac地址转换为六个字节,网上找了找资料,基本的思路就是将mac地址分为六段, 每段看做一个十六进制数,转化为十进制数后,分配赋值给六字节数组的每一位。后来在查看库函数的时候,发现了strtoul
函数,发现这个函数正好可以用来实现这个功能。实现函数如下:
int mac_str_to_bin( char *str, unsigned char *mac)
{
int i;
char *s, *e=NULL;
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";
unsignedchar 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 strtoul(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;
}
上面的方法解决了将MAC地址转换成数组的问题,但是我又遇到了新的问题,那就是在vs2015中的mac地址输入框中获取的mac地址是 CString
类型,但是上面的函数参数是 char*
类型,因此,还得先把CString
类型的mac地址转换成char*
类型,于是我查找了一些资料,发现一个比较好的函数,代码如下:
/**
* CString to char
*/
char * cs2ca(CString str)
{
char *ptr;
#ifdef _UNICODE
LONG len;
len = WideCharToMultiByte(CP_ACP, 0, str, -1, NULL, 0, NULL, NULL);
ptr = new char[len + 1];
memset(ptr, 0, len + 1);
WideCharToMultiByte(CP_ACP, 0, str, -1, ptr, len + 1, NULL, NULL);
#else
ptr = new char[str.GetAllocLength() + 1];
sprintf(ptr, _T("%s"), str);
#endif
return ptr;
}