inet_addr函数

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);
}


你可能感兴趣的:(inet_addr函数)