inet_addr函数
今天使用了在编程时使用到inet_addr函数,不太熟悉,找了点资料,整理如下。
inet_addr的功能是将一个ip地址字符串转换成一个整数值。一般的IP地址串格式为:'a.b.c.d'分成四段。
但是也会有分为1、2或3段的格式。下面我们来看看WIN2K下inet_addr函数的源码:
unsigned long PASCAL inet_addr( IN const char *cp ) { register unsigned long val, base, n; register char c; unsigned long parts[4], *pp = parts; WS_ENTER( "inet_addr", (PVOID)cp, NULL, NULL, NULL ); again: /* * Collect number up to ``.''. * Values are specified as for C: * 0x=hex, 0=octal, other=decimal. */ val = 0; base = 10; //每段中根据中根据前缀知道其使用的进制。 if (*cp == '0') { //0X前缀为十六进制;0前缀为八进制;缺省为十进制 base = 8, cp++; if (*cp == 'x' || *cp == 'X') base = 16, cp++; } //下面这段代码是用来取得IP地址的一段将其转换值整数 while (c = *cp) { if (isdigit(c)) { //若为数字,则将其转换至数字进行计算 val = (val * base) + (c - '0'); //注意,在W2K下这儿有一个BUG,因为当为8进制时没有 cp++; //检查是否为8、9数字;这个问题在XP下修正了。 continue; } if (base == 16 && isxdigit(c)) { //当为16进制时,检查是否为‘a’~ 'f'字母 val = (val << 4) + (c + 10 - (islower(c) ? 'a' : 'A')); cp++; continue; } break; } //当为‘.’时,此时表示地址段结束了。 if (*cp == '.') { /* * Internet format: * a.b.c.d * a.b.c (with c treated as 16-bits) * a.b (with b treated as 24 bits) */ /* GSS - next line was corrected on 8/5/89, was 'parts + 4' */ if (pp >= parts + 3) { //若为‘.’间隔,检查是否已经超过3个了。 WS_EXIT( "inet_addr", -1, TRUE );//因为最多为3个‘.’ return ((unsigned long) -1); //在W2K下可以看出允许直接以‘.’开头的IP地址 } //而在XP下是不允许的。 *pp++ = val, cp++; goto again; } /* * Check for trailing characters.//到此处时,要么结束,要么为出错字符 */ //只有为0或者空格是结束,否则出错。 if (*cp && !isspace(*cp)) { WS_EXIT( "inet_addr", -1, TRUE ); return (INADDR_NONE); } *pp++ = val; /* * Concoct the address according to * the number of parts specified. */ n = (unsigned long)(pp - parts); switch ((int) n) { case 1: /* a -- 32 bits */ //当为一个段a时,直接就是这个值。 val = parts[0]; break; case 2: /* a.b -- 8.24 bits */ //当为两个段a.b时,整数值构成XX XX XX XX if ((parts[0] > 0xff) || (parts[1] > 0xffffff)) { // -- -------- WS_EXIT( "inet_addr", -1, TRUE ); // a b return(INADDR_NONE); } val = (parts[0] << 24) | (parts[1] & 0xffffff); break; case 3: /* a.b.c -- 8.8.16 bits *///当为3个段a.b.c时,整数值构成XX XX XX XX if ((parts[0] > 0xff) || (parts[1] > 0xff) || // -- -- ----- (parts[2] > 0xffff)) { // a b c WS_EXIT( "inet_addr", -1, TRUE ); return(INADDR_NONE); } val = (parts[0] << 24) | ((parts[1] & 0xff) << 16) | (parts[2] & 0xffff); break; case 4: /* a.b.c.d -- 8.8.8.8 bits *///当为4个段a.b.c.d时,整数值构成XX XX XX XX if ((parts[0] > 0xff) || (parts[1] > 0xff) || // -- -- -- -- (parts[2] > 0xff) || (parts[3] > 0xff)) { // a b c d WS_EXIT( "inet_addr", -1, TRUE ); return(INADDR_NONE); } val = (parts[0] << 24) | ((parts[1] & 0xff) << 16) | ((parts[2] & 0xff) << 8) | (parts[3] & 0xff); break; default: WS_EXIT( "inet_addr", -1, TRUE ); return (INADDR_NONE); } val = htonl(val); WS_EXIT( "inet_addr", val, FALSE ); return (val); }