mac地址转换为6字节数组--strtoul函数

最近工作中需要在网络中传递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编写的。

你可能感兴趣的:(库函数拾遗)